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内 容 提 要 





























本 书 以 通俗 易 懂 的 语言 向 初学 者 介绍 了 PHP 语言 的 基本 概念 、 使 用 方法 和 注意 事项 。 全 书 通过 丰富 
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了 中 








2000 年 我 在 编写 本 书 第 一 版 时 ，PHP 还 只 是 一 个 几乎 不 为 人 知 的 开源 项 目 。 它 被 熟知 内 幕 的 
技术 人 员 所 钟爱 ， 但 是 还 没有 像 今天 这 样 成 为 Web 开 发 方面 公认 流行 的 选择 。 在 我 自学 PHP 的 时 
候 ， 关 于 这 种 语言 的 文档 少 之 又 少 ， 这 正 是 我 编写 此 书 的 初衷。 

现在 形势 不 一 样 了 。 因 特 网 经 历 大 起 大 落 之 后 ， 进 入 了 稳定 的 发 展期 。 而 且 ，PHP 现 在 已 经 
成 为 程序 员 首 选 的 动态 Web 设 计 工具 ， 并 且 开 始 将 领域 扩张 至 Web 开 发 之 外 。 但 是 ， 尽 管 PHP 已 
经 广 为 流 行 ， 相 关 的 文档 、 示 例 代 码 和 样 例 也 越 来 越 丰富 ， 出 版 一 本 介绍 PHP 的 好 书 还 是 很 有 必 
要 的 。 PHP 已 经 发 布 了 5 个 主要 版 本 , 对 于 学 习 PHP 的 读者 来 说 ,本 书 这 样 简洁 实用 的 教程 正 是 所 
需 的 最 佳 指 南 。 

本 书 不 仅 帮助 读者 次 和 入 理解 基本 原理 ， 同 时 还 将 引导 读者 获取 进 阶 信息 。 尽 管 本 书 不 是 编程 
参考 大 全 , 但 是 通过 详尽 阐述 和 真实 样 例 ， 为 读者 提供 了 使 用 PHP 构 建 动态 Web 站 点 和 Web 应 用 
程序 的 必 备 知识 。 


PHP 是 什么 

















PHP 起 初 是 Personal Home Page 的 缩 
写 ， 意 为 个 人 主页 。 它 最 早 是 由 Rasmus 
Lerdorf 在 1994 年 创建 的 ， 用 来 跟踪 他 本 
人 在 线 简历 的 访问 者 。 随 着 PHP 的 实用 性 
和 功能 的 扩展 (同时 它 被 应 用 在 越 来 越 专 
业 的 场景 中 ), 它 代表 的 意思 变化 为 PHP: 
Hypertext Preprocessor (PHP， 超 文本 预 
处 理 程序 ) 。[ 这 个 定义 的 主要 意思 是 PHP 
在 数据 变 为 HIML (HyperText Markup 
Language) 之 前 先 处 理 数 据 。] 

通过 PHP 的 官方 网 站 www.php.net ( 参 
全 图 了解 中， 大 了。 图 上 该 图 为 编写 本 书 时 PHP 定 方 网 站 的 截图 ， 网 址 
内 了 式 觅 本 语言。 下 面 详细 解释 这 个 定义 a ee 
的 含义 。 足 对 它 好 奇 心 的 首选 地 址 


tho ret 
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What is PHP? 
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PHP 定 义 中 “HTML 内 嵌 式 ”的 意思 是 它 可 以 混杂 在 HTML 代 码 中 。HTML 是 一 种 用 来 生成 
所 有 Web 页 面 的 代码 。 因 此 ， 使 用 PHP 编 写 代 码 只 比 使 用 HTML 稍 微 复杂 一 点 点 。 

此 外 ， 相 对 于 编译 语言 来 说 ，PHP 是 一 种 脚本 语言 。 也 就 是 说 PHP 被 设计 成 为 仅 当 事件 发 生 
之 后 才 会 开始 工作 。 例 如 在 用 户 提交 表单 后 ， 或 者 前 往 某 个 URL (Uniform Resource Locator， 统 
一 资源 定位 符 ， 即 Web 地 址 ) 时 ，PHP 才 会 开始 工作 。 男 一 种 流行 的 脚本 语言 就 是 JavaScript， 它 
被 普遍 应 用 于 处 理 Web 浏 览 器 中 发 生 的 事件 。 这 两 种 语言 都 被 描述 成 解释 型 语言 ， 因 为 它们 的 代 
码 都 要 通过 一 个 可 执行 文件 (比如 PHP 解 释 器 或 浏览 器 的 JavaScript 引 擎 ) 来 解释 执行 。 相反， 像 
C 和 C++ 那样 的 编译 语言 可 以 用 来 编写 独立 的 应 用 程序 ， 编 译 后 可 直接 运行 。 

PHP 还 是 一 种 服务 器 端 技术 。 这 是 指 PHP 所 做 的 一 切 均 在 服务 器 端 (而 不 是 在 客户 端 一 一 用 
户 浏 览 网 站 的 计算 机 中 ) 发 生 。 服 务 器 是 一 台 计 算 机 ， 用 来 提供 用 户 使 用 浏览 器 (例如 ，Firefox、 
IE 或 者 Safari) 访问 某 个 Web 地 址 时 的 页 面 。 后 文 将 介绍 这 个 过 程 的 细 闻 (参看 “PHP 是 如 何 工作 
的 ”一 市 )。 

最 后 ，PHP 是 一 种 跨 平 台 的 技术 ， 也 就 是 说 它 能 够 用 在 运行 Unix、Windows、Macintosh 和 其 

他 操作 系统 的 机 器 上 。 当然 我 们 讨论 的 是 服务 器 运行 的 操作 系统 , 而 不 是 客户 端 运行 的 操作 系统 。 
PHP 不 仅 能 够 运行 在 几乎 所 有 的 操作 系统 上 ， 而 且 与 其 他 大 多 数 的 编程 语言 不 同 ， 它 能 够 在 不 同 
的 操作 平台 上 进行 切换 ， 而 不 做 或 者 仅仅 做 很 少 的 修改 。 
在 撰写 本 书 时 ，PHP 的 版 本 为 5.3.5 和 5.2.17 (5.3 和 5.2 两 个 版 本 之 间 差 别 不 是 很 大 ， 因 此 在 一 
段 时 间 内 5.2 还 将 获得 支持 )。 尽 管 本 书 的 代码 环境 使 用 稳定 的 PHP 5.3, 但 是 所 有 的 代码 都 能 够 向 
后 兼容 。 如 果 不 能 兼容 4.x 版 ， 至 少 能 够 支持 PHP 的 5.x 版 。 本 书 偶尔 会 使 用 最 近 版 本 的 PHP， 可 能 
会 与 旧版 本 有 些许 不 同 ， 此 时 会 在 注解 或 者 提示 中 说 明 如 何 相应 地 调整 代码 。 

请 访问 PHPnet 和 www.zend.com 以 获得 更 多 信息 ，zend.com 体 现 了 PHP 的 核心 思想 (参见 
图 i-2 ) 。 
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图 i-2 ”这 是 Zend 的 主页 , 聚集 了 编写 PHP 的 核心 开发 人 员 。 该 网 站 
包含 了 大 量 有 用 的 软件 、 代 码 库 和 编写 得 很 好 的 教程 








PHP 的 局 限 性 


UOUPHEUUUOUOUOOOOOOOOUOUOPHEUOUDOOUOOUOOUOUOUU0OO0ODO 
各 加 加 

DOJavascip OOOO000000000000000000000000U0U000 
半 加 加 冯 加 
UOUUD0UPHPUDOOOOOOUPHPUOUOUOOUOOOOOOOOUOUU0O0O0000 艺 
DUUUDDPHP LU JavascriptUUUOUPHPUUHIMIODD 

DUOUUUUPHPIUUOUUUOUUUUPHRPIY webUUUOUUOUOHIMOU NNO 
DO000000000000000000000000000DU00U WedU0UU00 
U0000 


为 什么 要 使 用 PHP 


简 而 言 之 ， 与 其 他 同类 语言 相 比 ，PHP 能 够 表现 得 更 好 、 更 快 ， 并 且 更 简单 易学 。 所 有 的 
Web 站 点 都 必须 以 HTML 开 始 ， 因 此 可 以 使 用 许多 静态 HTML 页 面 来 创建 一 个 完整 的 站 点 。 但 是 
基础 的 HTML 在 灵活 性 和 提供 响应 方面 都 有 局 限 。 访 问 者 进入 HTML 页 面 时 看 到 的 是 简单 的 、 没 
有 定制 或 者 动态 行为 的 页 面 。 使 用 PHP 则 可 以 进行 同 数据 库 和 文件 的 交互 、 处 理 邮 件 等 操作 ， 做 
很 多 HTML 不 能 做 的 事情 。 

很 久 以 前 ，Web 站 点 设计 者 就 认识 到 不 能 单独 用 HIML 创 造 出 迷人 、 持 和 久 的 Web 站 点 。 为 了 
结束 这 种 状态 ， 诸 如 PHP 的 服务 器 端 技术 应 运 而 生 。 这 些 技术 能 够 让 Web 页 面 设计 者 创建 Web 应 
用 程序 ， 这 些 Web 应 用 程序 能 够 动态 地 生成 ， 并 且 将 程序 员 所 渴望 的 元 素 都 纳入 考虑 的 范畴 。 这 
些 高 级 站 点 通常 都 是 数据 库 驱 动 的 ， 能 够 比 静 态 HTML 页 面 更 快 地 升级 和 维护 。 

当选 择 服 务 器 端 技术 时 ，PHP 最 主要 的 对 手 是 CGI 脚本 (Common Gateway Interface， 通 用 网 
关 接 口 ， 并 不 一 定 用 Perl 编 写 )、ASPNET、Adobe 的 ColdFusion、JSP (JavaServer Pages) 和 Ruby 
on Rails。 因 为 JavaScript 是 一 种 客户 端 技 术 , 并 且 不 能 同 PHP 或 其 他 这 类 技术 一 样 用 来 创建 HTML 
页 面 ， 所 以 JavaScript 并 不 是 PHP 的 真正 替代 品 (反之 亦 然 )。 

现在 问题 是 , 为 什么 Web 设 计 者 要 用 PHP 而 不 用 CGI、ASPNET、JSP 等 技术 来 创建 动态 的 Web 
站 点 呢 ? 

口 PHP 更 易于 学 习 和 使 用 。 在 阅读 本 书 之 后 ， 没 有 经 过 正式 编程 培训 的 人 都 能 够 轻松 地 编 

写 PHP 脚 本 。 相 比较 而 言 , ASP.NET 要 求 了 解 VBScript、C# 或 者 其 他 语言 CGI 要 求 有 Perl 
(或 者 C) 编程 基础 。 它 们 都 是 更 加 复杂 且 难 以 学 习 的 语言 。 

口 PHP 专 门 用 来 编写 动态 Web 页 面 。Perl (以 及 VBScript 和 Java) 则 不 是 ， 这 个 事实 暗示 了 
出 于 自身 特定 的 意图 ,PHP 能 够 在 处 理 特 定 任务 时 比 它 的 那些 竞争 者 更 加 迅速 .更 加 简便 。 
但 是 本 书 需 要 点 明 的 是 ， 虽 然 在 处 理 某 些 任务 时 更 加 优秀 (因为 它 就 是 为 了 解决 这 些 问 
题 而 创造 出 来 的 )，PHP 并 不 是 比 Java 或 者 Perl 更 好 的 编程 语言 一 一 后 两 者 能 够 做 许多 PHP 
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不 能 做 的 事情 。 

口 PHP 不 仅 免 费 而 且 跨 平台 。 因 此 ， 可 以 在 任何 计算 机 上 使 用 它 而 不 产生 费用 。 此 外 ，PHP 

的 开源 本 质 上 意味 着 是 PHP 的 用 户 推 动 了 它 的 发 展 ， 而 不 是 由 某 个 企业 实体 推动 的 。 

口 PHP 是 可 以 用 来 开发 动态 Web 站 点 的 最 受 欢迎 的 工具 。 在 编写 本 书 时 ， 有 超过 75% 的 站 
点 是 由 PHP 编 写 的 (参见 图 i-2), 在 所 有 的 编程 语言 中 , 受 欢 迎 程度 上 , PHP 排 在 第 4 位 ( 参 
见 图 i-3)。 很 多 大 型 Web 网 站 (如 Yahoo!、Wikipedia 和 Facebook 等 ) 和 内 容 管 理工 具 (如 
WordPress、Drupal、Moodle 和 Joomla) 都 使 用 PHP。 和 掌握 这 项 技术 ,你 不 仅 发 展 了 一 项 实 
用 的 业余 爱好 ， 同 时 也 掌握 了 一 门 可 使 自己 获 利 的 技能 
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图 i-3 Tiobe Index (http:/www.tiobe.com/index.php/content/paperinfo/tpci/index.html ) 
对 流行 的 编程 语言 的 综合 排名 。 


PHP 是 如 何 工作 的 


PHP 是 一 种 服务 器 端 语言 ， 这 意味 着 用 PHP 编 写 的 代码 将 在 为 Web 浏 览 器 提供 Web 页 面 的 主 
机 上 运行 。 当 访问 一 个 Web 站 点 (例如 ，www.LarryUllman.com) 时 ， 所 涉及 的 Internet 服 务 提供 
者 (ISP) 将 把 请 求 定 向 到 保存 着 www.LarryUllman.com 信 息 的 服务 器 上 。 服 务 器 读 取 PHP 代 码 并 
执行 脚本 指令 。 在 这 个 示例 中 ，PHP 代 码 告知 服务 器 以 HTML 的 形式 向 浏览 器 发 送 适 当 的 Web 页 
面 (参见 图 14)， 简 而 言 之 ，PHP 按 照 所 选择 的 参数 创建 了 一 个 HTML 页 面 。 

与 HTML 生 成 的 网 站 有 所 不 同 ， 当 请 求 发 出 时 , 服务 器 仅仅 向 Web 浏 览 器 发 送 HTML 数 据 一 一 
没有 服务 器 端 解释 发 生 (参见 图 i-5)。 换 句 话说 ， 在 最 终 用 户 的 浏览 器 上 查看 home.html 和 
home.php 并 不 一 定 有 明显 的 区 别 ， 但 是 如 何 生成 这 两 个 页 面 却 有 很 大 不 同 。 主 要 的 不 同 之 处 在 
于 ， 使 用 PHP 可 以 让 服务 器 动态 地 生成 HTML 人 代码。 例如， 今天 是 星期 一 而 不 是 星期 二 ， 或 者 如 
果 用 户 已 经 访问 过 该 页 ， 这 样 的 不 同 信息 能 够 呈现 出 来 。 动 态 Web 页 面 的 创建 ， 将 不 那么 吸引 人 
的 静态 网 站 同 更 有 趣 因 而 访问 量 更 大 、 更 具有 交互 性 的 网 站 区 分 开 来 。 
































客户 端 URL 请 求 服务 器 
一 ”PE 下 


一 | 


图 i-4 这 张 图 示范 了 在 客户 端 和 服务 器 端 之 间 进 行 的 处 理工 作 (虽然 是 在 极其 简单 的 条 
件 下 )，PHP 模 块 ( 添 加 进 服务 器 的 一 个 应 用 程序 ， 用 来 增加 它 的 功能 ) 用 来 向 浏 
览 器 发 回 HTML。 所 有 的 服务 器 端 技术 都 在 服务 器 上 使 用 一 个 第 三 方 模块 ， 用 来 

































































处 理发 送 回 客户 端的 数据 
客户 端 URL 请 求 服务 器 
HTML 
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i-5 ”将 图 i-5 中 服务 器 处 理 HTML 的 方式 与 图 i-4 进 行 比较 。 这 跟 通 过 浏览 器 查看 本 地 
的 HTML 页 面 没 有 区 别 一 一 本 地 页 面 不 需要 服务 器 处 理 , 但 是 动态 生成 的 页 面 需 
要 通过 服务 器 访问 ， 因 为 动态 页 面 要 经 过 服务 器 处 理 

















使 用 PHP 和 直接 使 用 HTML 之 间 重 要 的 不 同 之 处 在 于 ，PHP 在 服务 器 端 处 理 完 所 有 的 事情 之 
后 向 浏览 器 发 送 适当 的 信息 。 本 书 将 介绍 如 何 使 用 PHP 向 浏览 器 发 送 正 确 的 数据 。 


准备 工作 


使 用 PHP 最 重要 的 条 件 是 能 够 访问 启用 了 PHP 的 服务 器 ， 这 是 因为 它 是 一 种 服务 器 端 脚 本 话 
言 。 由 于 PHP 相 当 普及 ,因此 ISP 或 者 Web 主 机 提供 商 极 有 可 能 已 经 在 他 们 的 服务 器 上 安装 了 PHP 
模块 。 使 用 时 需要 同 他 们 联系 以 确认 支持 哪些 技术 。 

另外 一 个 选择 是 , 在 自己 的 计算 机 上 安装 PHP 和 Web 服 务 器 应 用 程序 (如 Apache) 。Windows、 
Mac OS X 或 者 Linux 的 用 户 都 能 够 很 容易 地 免费 安装 和 使 用 PHP。 附 录 人 A 提供 了 安装 PHP 的 指导 。 
如 果 和 希望 使 用 自己 的 服务 器 ， 并 且 自 己 安装 PHP， 那 么 可 以 在 PHP 的 Web 站 点 (www.php.net) 上 
找到 免费 下 载 的 安装 包 。 如 果 采 用 后 一 种 方式 (这 也 是 我 们 推荐 的 方式 )， 那 么 你 的 计算 机 将 可 
以 同时 作为 客户 端 和 服务 器 端 使 用 。 

第 二 个 条 件 是 ， 在 计算 机 上 必须 拥有 一 个 文本 编辑 器 。Crimson Editor、SciTE、TextWrangler 
以 及 类 似 的 免费 应 用 程序 都 能 够 很 好 地 满足 需要 ， 同 时 BBEdit、EditPad、TextMate 和 其 他 的 商业 
应 用 程序 可 以 提供 用 户 喜 欢 的 更 丰富 的 特性 。 如 果 习 惯 使 用 如 Adobe Dreamweaver (参见 图 1-6) 
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或 Aptana Studio 的 图 形 化 界面 (也 被 称 作 WYSIWYG 一 一 所 见 即 所 得 )， 


序 的 手册 以 了 解 如 何 利用 它们 进行 编程 。 


可 以 考虑 查阅 这 些 应 用 程 










































































DO 
Open a Recent ltem Create New Top Features (videos) 
二 Untitled-z.html 富 HTML 中 Related Files 
人 Untitled-1.html 人 coldFusion 一 
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图 -6 广 受 





欢迎 的 Dreamweaver IDE 提 供 包括 PHP 在 内 的 服务 器 端 技 术 的 开发 环境 


第 三 ，PHP 是 一 种 把 在 文本 编辑 器 中 编写 的 脚本 放 到 服务 器 上 的 方法 。 如 果 在 自己 的 计算 机 
上 安装 了 PHP， 那 么 可 以 将 它们 保存 在 适当 的 目录 中 。 但 是 ， 如 果 使 用 ISP 或 者 Web 主 机 提供 商 提 
供 的 远程 服务 器 ， 那 么 需要 一 个 FTP (File Transfer Protocol， 文件 传输 协议 ) 应 用 程序 把 脚本 上 传 
到 服务 器 。 在 第 1 章 中 ,本 书 将 使 用 免费 的 FileZilla (www.filezilla-project.org， 参见 图 i-7) 作为 示例 。 
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图 -7 FileZilla 应 用 程序 可 
和 其 他 的 文件 


行 在 许多 操作 系统 上 ， 用 于 向 远程 服务 器 上 传 PHP 脚 本 
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最 后 ， 如 果 读 者 想 试验 第 12 章 中 的 示例 ， 那 么 使 用 MySQL (www.mysql.com， 参见 图 -8) 或 
者 其 他 数据 库 应 用 程序 。MySQL 也 是 免费 的 应 用 程序 ， 可 以 安装 在 自己 的 计算 机 上 。 

















图 -8 (在 编写 此 书 时 ) MySQL 的 Web 站 点 


本 书 假定 读者 仅 具 备 HTML 的 基础 知识 。 当 然 ， 如 果 比 较 熟 悉 手 工 编 写 HTML 代 码 ， 而 不 是 
使 用 诸如 Dreamweaver 这 类 创建 Web 网 站 的 辅助 应 用 程序 ， 那 么 学 习 PHP 将 变 得 更 加 容易 。 每 个 
程序 员 都 会 需要 查找 HTML 参 考 资 料 ， 不 论 之 前 已 经 了 解 多 少 相关 知识 ,我 都 建议 你 将 一 本 好 的 
HTML 参考 书 放 在 手边 。 这 类 HTML 编 程 入 门 书籍 有 Elizabeth Castro 编 著 的 《HTML XHTML CSS 
基础 教程 (第 6 版 )》( 人 民 邮 电 出 版 社 2007 年 出 版 )。 

本 书 并 不 要 求 读者 具备 编程 的 经 验 。 但是， 如 果 你 有 编程 经 验 将 会 加 速 学 习 的 进度 ， 因 为 在 
学 习 的 过 程 中 你 会 很 快 发 现 有 很 多 地 方 都 同 以 往 的 经 验 相 通 。 例 如 ，Perl 和 PHP 或 者 JavaScript 和 
PHP 都 有 类 似 之 处 。 
关于 本 书 

本 书 旨 在 向 读者 介绍 PHP 编 程 的 基础 知识 ,同时 也 对 他 们 将 来 可 能 用 到 的 更 多 高 级 特性 给 出 
提示 ， 而 没有 过 分 探究 细节 。 本 书 中 的 内 容 将 延续 下 面 的 惯例 。 

逐步 介绍 部 分 指明 了 需要 你 手工 编写 的 代码 , 以 及 该 代码 在 脚本 中 的 位 置 。 这 些 需 要 输入 的 
代码 以 下 面 字 体 给 出 。 例 如 : 


<?php print "Hello, World! "; ?> 








本 版 的 新 内 容 
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0000PHP60 MySQLS5O OOOO00 00000000000000000000 
加 
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回回 加 轩 加 加 加 加 时时 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 加 轩 
D00000000000000000000 www.LaryUliman.com/forom/l] 
可 OTDOTOYOOGUOOYU 


PHP 代 码 还 会 以 完整 的 脚本 给 出 ， 并 且 带 有 行 号 作为 参考 (参看 脚本 i-1)。 读 者 在 编写 代码 
i 因为 这 样 将 导致 代码 不 能 执行 。 本 书 建议 使 用 能 自动 显示 行 号 的 文本 编辑 

















， 这 些 数字 可 以 提供 帮助 。 有 时 候 ， 你 会 发 现 脚 本 中 某 些 特殊 的 行 





二 0 | 起 你 :对 新 的 或 者 其 他 相关 信 息 的 注意 。 


脚本 i-1 
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PHP 示 例 脚本 ， 标 记 有 行 号 ， 并 且 对 特殊 代码 段 用 粗 体 强调 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Hello, World!</title> 

</head> 

<body> 

<?php print "Hello, World!"; ?> 

</body> 

</html> 


根据 PHP 的 工作 原理 ， 每 段 脚本 都 可 以 从 3 个 角度 来 观察 : PHP 代 码 (如 脚本 i-1)、 发 送 给 浏 
览 器 的 代码 (主要 是 HTML) 以 及 浏览 器 向 最 终 用 户 的 呈现 效果 。 在 适当 情况 下 ， 本 书 将 通过 部 
分 或 完整 的 浏览 器 窗口 的 屏幕 截图 ， 呈 现 脚本 示例 运行 的 最 终结 果 (参见 图 -9)。 有 时 候 也 会 给 
出 浏览 器 显示 接收 到 的 HTML 源 代码 的 截图 (参见 图 i-10)。 通常 可 以 在 适当 的 Web 浏 览 器 菜单 中 ， 
选择 View Source 或 者 View Page Source 来 查看 源 代 码 。 总 之 , 图 1-10 显 示 了 浏览 器 接收 到 的 HTML， 






































而 图 i-9 演 示 了 浏览 器 解释 相应 HTML 的 结果 。 使 用 PHP， 可 以 创建 发 送 到 浏览 器 的 HTML。 


{人 Hello, World! - Windows Internet Ex... 国 | 回 | 
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图 -9 ”这 是 一 个 示例 视图 ， 可 以 看 到 浏览 器 窗口 。 对 于 本 书 中 的 示例 ， 
无 论 你 使 用 什么 浏览 器 或 操作 系统 ， 结 果 都 不 会 有 什么 不 同 














@ 本 











已 由 人 民 邮 电 出 版 社 于 2008 年 10 月 出 版 。 一 一 编者 注 























| http:1710.0.1.2:8888/script_i_01.php - Original Source 
le Edit Format 
> <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
hetp: YUWWUW.W3 .0rg/TR/ Xhtml1l/DTD/ xhtmll-transitional.dtd"> 
5 <html xmlns="http://Wwwy.w3.org/1999/xhtml" xml:lang="en" lang="en"> 
<head> 





<meta http-edqu "Content-Type" content="text/html; charset=utf-8"/> 
<title>Hello i rld!i</title> 


Hello, World!'</body> 
</html> 











图 i-10 ”查看 Web 浏 览 器 获取 的 源 代码 ， 可 以 看 到 由 服务 器 发 送 的 PHP 创 建 的 HTML 


由 于 版 面 有 限 ， 因 此 逐步 介绍 中 的 某 些 PHP 代 码 行 在 印刷 时 会 出 现 折 行 ， 这 种 情况 也 许 在 读 
者 的 编辑 器 中 不 会 出 现 。 在 出 现 这 样 的 情况 时 ， 本 书 会 用 一 个 小 灰色 箭头 表示 折 行 。 例如 : 


print "This is going to be a longer line 
of code."; 











对 此 , 读者 在 脚本 中 的 代码 应 该 还 是 写 在 一 行 中 ,否则 在 执行 时 将 会 遇 到 错误 ( 编 有 行 号 的 
脚本 中 不 使 用 灰色 箭头 ) 。 

当 示范 新 特性 和 新 技术 时 ， 本 书 将 尽力 解释 清楚 “为 什么 这 样 ” 以 及 “如 何 做 到 “。 通 过 阅读 
理解 和 使 用 国 数 ， 应 该 可 以 确切 地 掌握 它 。 当 然 ， 有 一 些 疑问 是 不 可 避免 的 ， 本 书 为 此 提供 了 一 些 
参考 资源 以 便 查找 到 问题 的 答案 (参看 附录 B) 。 如 果 对 某 些 特定 的 函数 或 者 示例 有 疑问 ， 最 好 查 
看 在 线 的 PHP 手 册 ， 或 者 参考 本 书 提供 的 Web 站 点 〈 以 及 其 中 的 用 户 支持 论坛 )。 























哪 本 书 更 适合 

加 IE 
D000000000000000000000000000000000000U000 
加 冯 呈 加 
加 加 

D0000000000000000000000000 W000UUUUUO 
DDO0000000000000000000UUUO0UPHPO MySQLSUUUUDUUD 
UUUUsSQU MySQLUUUOUOOUUOUOO0OOOOUOUO0OUOU0O0O0U000D 


关联 网 站 


本 书 关联 的 Web 站 点 非常 有 用 ， 其 网 址 是 www.LarryUlliman.com。 在 这 里 ， 可 以 下 载 书 中 涉 
及 的 每 段 代码 ”。( 但 是 ， 本 书 强烈 建议 读者 自己 输入 这 些 代码 ， 这 样 对 熟悉 PHP 的 结构 和 语法 都 
非常 有 帮助 。) 

这 个 网 站 还 包括 一 个 更 加 详细 的 参考 部 分 ， 这 个 部 分 提供 了 大 量 Web 页 面 的 链接 ， 这 些 











Q@ 本 书 代 码 也 可 以 从 图 灵 网 站 www.turingbook.com 本 书 网 页 免费 注册 下 载 。 一 一 编者 注 
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页 面 可 以 进一步 学 习 PHP。 此 外 ， 站 点 的 勘误 部 分 还 列 出 了 本 书 中 存在 的 错误 。 
本 书 的 支持 论坛 可 能 会 给 读者 带 来 最 多 帮助 ， 论 坛 的 地 址 是 www.LarryUllman.com/forum。 
这 个 论坛 可 以 帮 你 : 
口 找到 问题 的 答案 ， 
口 得 到 如 何 实 现 你 的 想法 的 建议 ， 
口 获得 来 自 其 他 读者 的 调试 帮助 ， 
口 了 解 这 项 技术 的 变化 给 本 书 中 的 示例 带 来 怎样 的 影响 ， 
口 学 习 其 他 人 如 何 使 用 PHP，; 
口 获得 复习 中 的 问题 答案 ; 
D 获得 比 直接 给 我 发 送 邮件 更 快 的 回复 。 











问题 、 意 见 和 建议 


如 果 有 与 PHP 相 关 的 问题 ， 可 以 通过 新 闻 组 、 邮 件 列 表 ， 以 及 PHP 有 关 的 Web 网 站 中 的 问答 
区 来 解决 。 附 录 B 更 详细 地 讨论 了 这 方面 的 内 容 。 浏 览 这 些 参考 资料 或 者 在 Internet 上 搜索 通常 是 
最 快 获得 问题 解答 的 方式 。 

也 可 以 将 问题 、 意 见 和 建议 直接 发 送 给 我 。 使 用 本 书 的 关联 论坛 ， 将 获得 最 快 的 答复 (我 通 
常会 首先 在 这 里 回答 问题 )。 如 果 更 希望 用 邮件 的 方式 ,我 的 联系 信息 在 Web 网 站 上 也 可 以 找到 。 
我 将 尽力 回复 收 到 的 每 封 邮 件 , 但 也 许 要 等 上 儿 周 (如 果 在 论坛 上 提问 ， 可 能 只 要 等 几 天 就 可 以 
收 到 回复 )。 

如 果 和 希望 获得 更 多 的 提示 和 启蒙 式 的 了 阅读， 请 在 www.catb.org/~esr/faqs/smart-questions.html 
上 参看 Eric Steven Raymond 所 写 的 文章 的 “How to Ask Questions the Smart Way”。 花 10 分 钟 来 阅 
读 该 文章 ， 可 以 更 快速 地 得 到 问题 的 答案 。 包 括 我 在 内 的 其 他 将 回答 问题 的 人 们 会 不 胜 感激 |! 
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本 章 内 容 

口 HTML 语 法 基础 

D PHP 语 法 基础 

口 使 用 FTP 

口 测试 脚本 

口 向 浏览 器 发 送 文本 
口 使 用 PHP 手 册 

口 向 浏览 器 发 送 HTML 
为 脚本 添加 注释 
口 调试 的 基本 步 又 
D 回顾 和 实践 





学 习 任何 新 的 编程 语言 时 ， 首 先 需 要 理解 的 是 它 的 基础 语法 ， 这 也 正 是 本 章 介绍 的 内 容 。 本 
书 将 重点 讨论 PHP 的 基础 知识 ， 但 是 同时 也 会 涉及 一 些 备 受 推崇 的 编程 技术 ， 从 长 远 来 看 ， 这 些 
技术 能 够 改进 我 们 的 工作 。 

如 果 读 者 之 前 没有 任何 编程 经 验 , 那 么 应 该 仔细 阅读 本 章 , 它 将 为 你 指明 今后 学 习 的 正确 方向 。 
如 果 已 经 具备 了 一 定 的 编程 经 验 , 那么 可 以 快速 浏览 这 一 章 , 了 解 本 书 其 余 的 内 容 。 在 结束 本 章 时 ， 
读者 就 将 能 够 成 功 地 编写 并 执行 第 一 个 PHP 脚 本 ， 走 上 开发 动态 Web 应 用 程序 之 路 。 


1.1 HTML 语法 基础 


所 有 的 Web 页 面 都 由 HTML (HyperText Markup Language， 超 文本 标记 语言 ) 构成 。 每 
种 Web 浏 览 器 (无论 是 微软 的 IE、 人 苹果 的 Safari、Mozilla 的 Firefox 还 是 Google 的 Chrome) 都 将 
HTML 代 码 : 


<hl>Hello, World!</hi1l> 
I just wanted to say <em>Hello</em>. 


转变 成 为 有 样式 的 Web 页 面 展 示 给 用 户 (参见 图 1-1)。 
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Hello, World! 


| just wanted to say Hello. 

















到 1-1 Web 浏览 器 呈现 HTML 代 码 的 方式 


在 撰写 本 书 时 , HTML 的 最 新 版 本 是 4.01。 下 一 个 主要 版 本 是 HTML5, 目前 正在 积极 地 研发 
和 讨论 中 ， 还 没有 最 后 定案 。 本 书 使 用 XHTMIL (eXtensible HTML)， 它 与 HTML 有 少许 差别 。 
实际 上 ，XHTML 同 HIML 非 常 相 似 ， 它 们 之 间 的 不 同 之 处 有 以 下 几 点 。 
口 所 有 的 标签 都 使 用 小 写字 母 。 
口 舱 套 标签 必须 有 恰当 的 格式 。 
这 条 规则 并 不 像 它 听 上 去 那么 复杂 ， 它 的 意思 是 ， 不 能 这 样 写 代码 : <div><p>text 
</div></p>， 出 应 该 使 用 这 种 格式 : <div><p>text</p></div>。 
口 所 有 的 标签 属性 值 必须 使 用 引号 引起 来 。 
在 HTML 中 可 以 这 样 写 : <table border=2>, 但 是 在 XHTML 中 必须 这 样 写 : <table 
border="2">。 
口 所 有 的 标签 必须 关闭 。 

这 条 规则 最 容易 让 大 多 数 人 感到 迷惑 。 许 多 HIML 标 签 同时 拥有 开启 和 关闭 标签 , 如 <div 
class="someclass">text</div>。 但 是 有 一 些 HTML 标 签 并 不 强制 使 用 关闭 标签 , 包 
括 <hr>、<br>、<img> 和 <input>。 为 了 编写 有 效 的 XHTML 标 签 ， 必 须 在 末尾 处 添加 
一 个 空格 和 和 斜 线 以 关闭 它们 ， 就 像 这 样 : 


<hr /> 

<br/> 

<img src="image.png" /> 

<input type="text" name="age" /> 























CSS 基 础 


HIMIN] xHIMLUOUODOOUOOOOOOOOUOUOOO0O0O0O0UO0O00000CSSS 
U Cascading Style SheetJ U DUOUOUODOUUOOUOOHIMCO XHIMLUOUOUOUOUU0OODO 
UUCssUUO0OU00O0O0OU0O0O00000CcSsSsuUUUOO0O0O0OCSsSSUOUOOO0ODODO 

UUUU0O0O0U0UYweouUOUUSssuUOO0OO0O0O0O0O0O0OU0UHIMLstylieDUDOD 


<style type="text/css"> 


CSS0 


</style> 


UUOUU0O0OU0O0OO0O0O0O0O0USSSOUODOOOOU0O0OU :inxUUOU0O00000 
U0U0USCSSUDD 


< ref te el ot veneee Ev De ene /> 


CssDOO0U0O00000000UCSSUU0UUUO0UD 
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img { border: 0px; } 

0 { i -| 

#about { background-color: #ccc; } 

昌 且 加 slassyg 0 
error[| 吕 中 0 


< Ma Vor Or EO /DS 


UUOUO0O0O0O00Uie00 abousUUOUOU0O00D0 


valour ANDOU /D> 


D0000000000is00000000000iad0U00D 
DO0U0O0000U0O0UCSSIUO0O00000000000000000000000 
DUUUDOO 

国人 
DD CSSO 


< ote voolor ed mor /> 


D0000000000000000U00CSSYUU0UUO 


在 深入 讲解 PHP 语 法 之 前 ， 让 我 们 创建 一 个 简单 但 是 有 效 的 XHTML 文 档 ， 它 将 作为 本 书 中 
几乎 所 有 示例 的 模板 。 

(1) 打开 文本 编辑 器 或 者 IDE。 

可 以 使 用 几乎 任何 应 用 程序 来 创建 HTIML、XHTML 和 PHP 页 面 。 通 常 的 选择 包括 有 : Adobe 
的 Dreamweaver (www.adobe.com) (运行 在 Windows 和 Mac OS X 上 )、EditPlus (www.editplus.com ) 
和 Crimson Editor (www.crimsoneditorcom) (运行 在 Windows 上 ), 以 及 运行 在 Mac 上 的 Bare Bones 
的 BBEdit (www.barebones.com) 或 者 MacroMates 的 TextMate (www.macromates.com ) 。 

(2) 选择 文件 > 新 建 来 创建 一 个 新 的 空白 文档 。 

一 些 文本 编辑 器 可 以 从 创建 一 个 某 种 类 型 的 新 文档 开始 ， 如 一 个 新 的 XHTML 文件 (参见 图 
1-2)。 如 果 所 使 用 的 应 用 程序 有 这 样 的 选项 ， 故 请 使 用 它 。 

(3) 从 XHTML 头 行 开 始 (参看 脚本 1-1): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 



































脚本 1-1 示例 文档 显示 了 基本 的 XHTML 代 码 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1l/DTD/xhtml1l-transitional.dtd"> 
<html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
<head> 

<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 








OPRODP 
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0 <br/> 
1 <p>Even with <span style="font-size: 150%;">some</span> decoration, it's still 


6 <title>Welcome to this Page!</title> 
7 </head> 

8 <body> 

9 <h1L>This is a basic XHTML page!</h1> 

站 

1 


not very exciting.</p> 
12 </body> 
13 </html> 





New HTML Document 





DD Insert XML Declaration 
Insert DOCTYPE | XHTML 1.0 Transitional 的 中 











Vi HTML Vi Head Yi Body 


| Give BBEdit Credit 








Title: ‘Untitled 























Lang: | 乓 Charset: 8 
Base: 及 
Meta: 

Link: 

Web Site: | Untitled Site | 全 | 

Template: | Default 峡 








图 1-2 BBEdit 和 其 他 大 多 数 Web 开 发 应 用 程序 一 样 ， 都 可 以 用 来 创建 基本 的 XHTML 文档 


一 个 有 效 的 XHTML 文档 都 以 这 几 行 开头 ， 它 们 用 来 告诉 Web 浏 览 器 文档 的 类 型 。 在 本 书 这 
个 模板 中 ， 将 创建 XHTML 1.0 过 渡 页 ， 这 就 是 说 将 遵循 XHTML 1.0 的 标准 。 过 没 部 分 的 意思 是 
这 里 将 允许 使 用 不 推荐 的 (不 再 被 建议 ) 标签 ( 同 严格 模式 相反 ， 在 严格 模式 中 这 是 不 允许 的 )。 

(4) 创建 页 面 的 head 部 分 : 


<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Welcome to this Page! </title> 

</head> 


XHTML 页 面 的 head 部 分 包含 content-type 元 标签 (有效 的 XHTML 所 必需 的 ) 和 title 
标签 。 框 注 “ 理 解 编码 ”讨论 了 标签 中 charset 部 分 的 意思 。JavaScript 和 CSS3 引 用 也 放 在 这 里 。 
(5) 创建 body 部 分 : 


<body> 

<hl>This is a basic XHTML page!</h1> 

<br/> 

<p>Even with <span style="font-size: 150%;">some</span> decoration, 
>it's still not very exciting.</p> 

</body> 
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在 Web 浏 览 器 中 所 看 到 的 页 面 的 内 容 包含 在 开始 和 关闭 的 body 标 签 之 间 。 遵 循 XHTML 的 规 J 
则 ， 换 行 标签 (<br/>) 包含 一 个 空格 和 紧 随 其 后 的 斜 线 ， 用 来 关闭 标签 。 所 有 的 其 他 标签 都 同 
它们 在 标准 HIML 中 对 应 的 标签 一 样 ， 只 不 过 要 使 用 小 写字 母 编写 。 其 中 的 CSS 代 码 用 来 增 大 
some 的 字号 。 

(6) 输入 </html> 以 完成 这 个 HTML 页 。 

(7) 选择 文件 > 另存 为 。 当 对 话 框 出 现时 ， 如 果 提 供 了 选项 ， 请 选择 TextOnly (或 者 ASCII) 
格式 。XHTML 和 PHP 文 档 都 是 纯 文本 文件 (这 同 Microsoft Word 文 档 专 有 的 格式 不 同 )。 在 保存 
文件 时 ， 还 可 能 需要 指明 编码 〈 请 参看 框 注 “理解 编码 ”) 。 

(8) 导航 至 希望 保存 脚本 的 位 置 。 

尽管 在 本 书 中 会 将 每 个 脚本 保存 在 各 自 专 有 的 文件 夹 里 , 但 是 读者 可 以 将 脚本 放置 在 计算 机 
上 的 任何 地 方 ， 当 然 为 每 章 建立 子 文件 夹 是 非常 必要 的 。 

(9) 将 文件 保存 为 velcome .html。 

虽然 这 里 是 使 用 XHTML 进行 编码 ， 但 是 页 
面 将 仍然 使 用 标准 的 .html1 或 者 .htm 扩 展 名 。 

(10) 在 web 浏览 器 上 查看 该 页 以 进行 测试 ”|This is a basic XHTML page! 
(参见 图 1-3 ) 。 

同 PHP 脚 本 不 同 (读者 很 快 就 能 发 现 )， 可 




















AMAO Welcome to this Pagel 


























以 通 过 在 Web 浏 览 器 中 吉 接 打 开 XHTML 和 Even with SOME decoration, it's still not Very exciting . 
HTML 页 面 测 试 它们 。 图 1-3 。 XHTML 文档 被 Web 浏 览 器 解释 显示 
理解 编码 


加 
加 
OODO 人 LO 
加 加 

DWeb O00000000000000000000 

<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 

charset=utf-8[| [00ODDUTF-S 8-bit Unicode Transformation Format[] 8[| Unicode 
UUOU0O0O000vnicodeUUOU00o0000000000000000 Unicoded 60 
DO00000000000000U0U090000000U00 Unicodel dU UTF-S0 

DOO00000000000 We ODOUTFS0U000000000000000000 
ET 
UUUUOOUIEUUOOUOOUUOOOUOOOUOUOU0OO0O0OYXeouOOOO0OuU000 
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V 提示 

口 请 访问 www.LarryUllman.com/forum/， 查 找 适 用 于 HTML 和 PHP 的 编辑 器 或 者 IDE。 

口 本 书 使 用 XHTML， 但 是 这 并 不 意味 着 你 也 必须 这 样 做 。 如 果 感 觉 HTML 更 加 顺手 ， 请 坚 

持 使 用 了 解 的 语言 。 这 样 不 会 影响 PHP 脚 本 运行 。 

口 参阅 Elizabeth Castro 的 优秀 著作 《HTML XHTML CSS 基 础 教程 (第 6 版 )》 以 了 解 更 多 关 

于 XHTML 、XML 和 HTML 的 信息 。 

口 本 书 将 混用 HIML 和 XHTML。 事 实 上 ， 大 多 数 时候 你 将 只 看 到 HIML， 但 是 请 注意 ， 实 
际 上 它 指 的 是 XHTML 。 


1.2 PHP 语法 基础 


在 了 解 了 如 何 编写 HTML 后 ， 那 么 现在 就 开始 介绍 PHP 脚 本 。 创 建 第 一 个 PHP 页 面 ， 我 们 完 
全 可 以 像 创 建 HTML 文 档 那 样 从 头 开始 做 起 。 请 记 住 下 面 这 个 至 关 重要 的 事项 : PHP 是 一 种 服务 
器 闯 技 术 ， 这 意味 着 它 不 能 在 客户 端 运行 ， 也 就 是 说 不 能 在 Web 浏 览 器 中 运行 。 但 是 Web 浏 览 器 
能 够 理解 HTML (以 及 JavaScript 和 CSS)， 因 此 PHP 是 用 来 生成 在 Web 浏 览 器 中 运行 的 HTML (请 
回顾 图 i-4 中 对 于 这 种 关系 的 图 示 )。 

标准 HTML 文 档 和 PHP 文 档 之 间 有 3 个 主要 的 不 同 之 处 。 首先 , 保存 PHP 脚 本 时 必须 使 用 .php 
扩展 名 (例如 ，index.php)。 其 次 ， 必 须 把 PHP 代 码 放置 在 <?php 和 ?> 标签 之 间 : 









































<body><h1l>This is HTML .</Ph1L> 
<?php PHP code! ?> 
<p>More HTML</p> 


PHP 标 签 指 明 页 面 中 的 这 一 部 分 将 由 PHP 执 行 。 这 将 引出 本 书 要 介绍 的 第 3 个 主要 的 不 同 之 
处 : PHP 脚 本 必须 运行 在 启用 了 PHP 的 Web 服 务 器 上 (而 HTML 页 面 可 以 在 任何 计算 机 中 浏览 )。 
这 就 是 说 PHP 脚 本 必须 总 是 通过 一 个 URL (如 http://something/page.php) 运行 。 如 果 在 Web 浏 览 器 
中 查看 PHP 脚 本 ， 地 址 前 应 该 加 上 http ， 否 则 PHP 脚 本 无 法 工作 。 

为 了 让 第 一 个 PHP 脚 本 做 一 些 事情 而 又 不 被 太 多 编程 过 程 所 烦恼 ， 这 里 将 使 用 phpinfo () 
函数 。 在 调用 这 个 函数 时 ,会 向 Web 浏 览 器 发 送 一 个 信息 表 。 这 个 表 列 举 了 安装 在 特定 服务 器 上 
的 PHP 的 具体 细节 。 这 是 检查 PHP 安 装 情况 的 一 种 非常 好 的 方法 ， 而 且 具 有 物 超 所 值 的 特质 。 














之 在 本 地 计算 机 上 创建 新 的 PHP 脚 本 








(1) 在 文本 编辑 器 或 者 IDE 中 创建 一 个 新 的 HTML 文 档 , 命名 为 phpinfo.php (参看 脚本 1-2): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
> "http://www.w3.0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>First PHP Script</title> 
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</head> 
<body> 
</body> 


</html> 


脚本 1-2 第 一 个 PHP 脚 本 使 用 一 个 典型 的 HTML 页 面 ， 添 加 PHP 标 签 ， 并 且 使 用 了 一 个 PHP 函 数 








1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML1.0 Transitional//EN" 

2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://ww.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 
5 

6 

7 

8 








<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>First PHP Script</title> 
</head> 
<body> 
9 <?php 
10 phpinfo(); 
11 ?> 
12 </body> 
13 </html> 


这 上 段 特殊 的 代码 在 很 大 程度 上 同 创建 PHP 页 面 没 有 关系 , 但 是 出 于 一 致 性 的 原因 ， 它 使 用 了 
和 基础 XHTML 示例 一 样 的 模板 (参看 脚本 1-1)。 

(2) 按 Retum 键 (苹果 机 ) 或 Enter 键 (PC) 以 在 开启 和 关闭 body 标 签 之 间 创 建 一 些 空白 行 。 

(3) 在 开启 body 标 签 后 的 那 行 输入 <?php。 

这 个 初始 PHP 标 签 告诉 服务 器 接 下 来 的 代码 是 PHP， 并 且 要 按照 PHP 进 行 处 理 。 

(4) 在 下 一 行 添加 下 面 的 内 容 : 

phpinfo(); 

本 书 随后 将 解释 该 语法 的 细节 ， 简 言 之 ， 这 是 调用 名 为 phpinfo 的 已 有 PHP 函 数 。 

必须 加 上 一 对 空 的 括号 “()”， 后 面 接 一 个 分 号 “;” 

(5) 在 关闭 body 标 签 之 前 的 那 行 输入 ?>。 

关闭 PHP 标 签 告知 服务 器 ， 脚 本 中 所 包含 的 PHP 部 分 已 经 结束 。 任 何在 PHP 标 签 之 外 的 文本 
都 将 立即 作为 HTML 发 送 给 Web 训 览 器 ， 并 且 不 会 被 当 作 PHP 代 码 来 处 理 。 

(6) 将 脚本 保存 为 phpinfo.php。 

不 想 过 分 强调 这 一 点 ， 但 是 请 记 住 PHP 脚 本 必须 使 用 有 效 的 文件 扩展 名 。 将 文件 保存 为 
filename.php 将 不 会 带 来 任何 问题 。 你 还 需要 确认 的 是 ， 应 用 程序 或 操作 系统 没有 为 文件 添加 
隐藏 的 文件 扩展 名 。 例 如 ，Windows 的 记事 本 会 为 不 常见 的 文件 扩展 名 添加 .Ext， 这 将 使 PHP 脚 
本 无 法 使 用 。 

ww 提示 

口 正如 计算 机 上 的 文件 扩展 名 会 告诉 操作 系统 使 用 哪 种 应 用 程序 来 打开 这 种 文件 一 样 ， 一 

个 Web 页 面 的 扩展 名 将 告诉 服务 器 如 何 处 理 相应 的 文件 : file.php 通 过 PHP 模 块 来 处 理 ， 
ASPNET 人 负责 处 理 file.aspx, file.html 则 是 静态 HTML 文 档 (通常 情况 下 )。 这 由 Web 
服务 器 的 应 用 程序 设置 而 决定 。 
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口 如 果 是 为 托管 的 Web 网 站 开发 PHP 脚 本 ， 请 向 托管 公司 查询 以 确认 可 以 为 PHP 文 档 使 用 哪 

种 文件 扩展 名 。 在 这 本 书 中 .php 将 是 最 常见 的 扩展 名 。 

口 读者 可 能 会 在 别人 的 脚本 中 偶尔 看 到 PHP 的 简写 标签 <? 和 ?> ， 但 是 最 好 坚持 使 用 正规 的 

标签 ， 正 如 此 书 中 所 示例 的 那样 。 实 际 上 ， 简 写 标 签 已 在 PHP 中 被 废弃 了 。 

口 为 phpinfo .php 文 件 创建 一 份 副本 会 很 方便 。phpinfo .php 脚 本 会 显示 出 当前 PHP 版 本 
支持 的 功能 、 当 前 设置 及 服务 器 的 其 他 功能 。 为 了 查看 这 些 信息 ， 本 书后 面 还 会 时 不 时 
地 提醒 你 运行 这 个 脚本 。 

口 不 使 用 Web 浏 览 器 ， 也 可 以 执行 PHP 脚 本 ， 但 这 需要 使 用 命令 行 界 面 和 一 个 独立 的 PHP 可 

执行 文件 。 这 个 主题 超过 了 本 书 讨论 的 范围 ， 而 且 这 种 方法 也 不 常用 。 


1.3 使 用 FTP 


与 HTML 能 够 在 自己 的 计算 机 上 进行 测试 不 同 ，PHP 脚 本 需要 在 启用 了 PHP 的 服务 器 上 运行 
以 查看 输出 结果 。PHP 通 过 Web 服 务 器 应 用 程序 来 运行 。 例如, Apache (http://httpd.apache.org)、 
Abyss (www.aprelium.com) 或 者 IIS (Internet Information Server, www.iis.net)。 

有 两 种 获取 PHP 服 务 器 的 方法 : 

(1) 在 自己 的 计算 机 上 安装 PHP 软 件 ; 

(2) 申请 网 络 托管 主机 。 

PHP 是 一 个 开源 软件 (从 某 种 程度 上 说 ， 它 是 免费 的 )， 安 装 很 容易 ， 也 不 会 对 计算 机 产生 
任何 负面 影响 。 如 果 需 要 安装 PHP 以 及 Web 服 务 器 ， 请 参考 附录 A。 如 果 已 经 完成 安装 ， 你 可 以 
跳 过 本 闻 ， 直 接 阅读 1.4 节 ， 了 解 如 何 测 试 你 的 第 一 个 PHP 脚 本 。 

如 果 没 有 在 本 机 运行 PHP， 那 么 需要 使 用 FIP 软 件 将 PHP 脚 本 传送 至 启用 了 PHP 的 服务 器 上 。 
Web 托 管 公司 或 者 服务 器 的 管理 员 将 提供 使 用 FTP 客 户 端 进行 访问 的 方式 。 有 很 多 FTP 客 户 端 可 
供 选 择 ， 在 下 面 的 一 系列 步骤 中 ， 本 书 将 选用 免费 的 FileZilla (http://filezilla-project.org/)， 它 可 
以 运行 在 许多 种 操作 系统 中 。 


信使 用 FTP 传 送 脚本 



























































(GD 打开 FTP 应 用 程序 。 
CO) 在 应 用 程序 的 连接 窗口 ， 输 入 Web 托 管 公司 所 提供 的 信息 (参见 图 1-4) 。FTP 访 问 需要 主 
机 名 或 下地 址 、 用 户 名 和 密码 。 














[有 二 | FileZilla 二 
A WR 
Host: www.larryullman | Username: someuser ] Password: eee | Port: | (CQuickconnect )™ 














1-4 ”在 Macintosh 中 显示 的 FileZilla 应 用 程序 主 窗口 


(3) 单 击 快速 连接 (或 者 在 你 的 FTP 客 户 端 中 进行 相应 的 操作 )。 
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否则 ， 将 在 FileZilla 窗 口 的 顶部 看 到 错误 消 | 本 二 





如 果 提 供 了 正确 的 信息 ， 那 么 就 可 以 进行 连接 。 
息 (参见 图 1-5) o 











0 


FileZilla 
3 


[| 




















Host: www.larryullman Username: someuser Password: se**eeeeeees Port ( Quickconnect J 
Command: USER someuser 
s 331 Password required for someuser 0 
和 A 
Critic e 
v 
、 、 口 于 
图 1-5 ”这 个 错误 报告 表明 登录 信息 不 正确 














(4) 导航 至 Web 页 面 的 正确 目录 (例如 ，www/ 或 者 htdocs/)。 

FTP 应 用 程序 并 不 会 导航 至 正确 的 目录 下 ， 这 里 需要 自己 导航 至 Web 文 档 根 目录 中 ， 这 个 目 
录 是 URL (例如 ，www.LarryUllman.com) 所 指向 的 服务 器 上 的 一 个 位 置 。 如 果 你 不 清楚 自己 电 
脑 的 Web 文 档 根 目录 ， 可 以 查看 主机 托管 商 提 供 的 说 明文 档 ， 或 向 他 们 寻求 帮助 。 

在 FileZilla 中 , 右 栏 显示 位 于 服务 器 上 的 文件 和 目录 , 左 栏 显示 位 于 本 机 上 的 文件 和 目录 ( 参 





















































见 图 1-6) o 双击 打开 文件 夹 。 
Local site: /Users/larryullman/Sites/ 后 Remote site: /httpdocs 8 
Tetures 看 con 
入 Public @ eror_docs @ 
pb ,ysites 全 bp jhttpdocs < 
bb Ctefflt > htnedare . 
Filename 八 Filesize Filetype Last modified Filename 八 Filesize Filetype Last modified 
DMCI Directory C55 Directory 10j0212010.. 
LarryUllman Directory downloads Directory 11/02/12010... 
PSU Directory 和 forum Directory 11/0212010.. 
YI Blog Directory images Directory 11/02/2010.. 
i Directory img Directory 1070812010 .. 
Yii1.1.3 Directory O7116/2010 】 js Directory 10131/12010.. 
图 1-6 “我 已 经 成 功 地 同 远程 服务 器 建立 了 连接 ， 并 且 导 航 至 httpdocs 目 录 (WebD DO0OD0OD0 ) 











(5) 将 脚本 phpinfo .php 上 传 至 服务 器 上 。 


使 用 FileZilla 上 传 脚本 ， 只 用 将 文件 从 左 栏 中 拖 中 至 右 栏 中 一 一 从 本 机 上 传 至 服务 


w 提示 


器 上 。 


口 一 些 文本 编辑 器 和 IDE 具 有 内 置 FIP 功 能 ， 可 以 直接 将 脚本 保存 到 服务 器 上 。 比 如 ， 
Dreamweaver 和 TextMate 不 用 退出 应 用 就 可 以 运行 PHP 脚 本 。 


1.4 


测试 脚本 
测试 脚本 分 为 两 步 。 


一 步 ， 将 PHP 脚 本 放 在 Web 服 务 器 的 正确 目录 下 。 


第 二 步 ， 在 Web 浏 
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览 器 上 加 载 正 确 的 URL 运 行 PHP 脚 本 。 

如 果 使 用 的 是 独立 的 Web 服 务 器 (如 托管 公司 提供 的 主机 )， 只 需要 使 用 FTP 应 用 程序 上 传 
PHP 脚 本 即 可 。 如 果 已 经 在 自己 的 计算 机 上 安装 了 PHP， 那 么 就 可 以 将 PHP 脚 本 保存 在 (或 者 
移动 至 ) Web 文 档 的 根 目 录 中 来 测试 脚本 。Web 文 档 的 根 目录 通常 表现 如 下 : 

口 ~/Sites，Mac OS X 用 户 (~ 代表 主 目 录 )， 

口 AbyssDir/htdocs， 在 所 有 操作 系统 中 ，AbyssDir 是 Abyss Web 服 务 器 的 安装 目录 ; 
口 C:\Inetpub\wwwroot， 运 行 IS 的 Windows 用 户 ， 

口 Cc:\xampp\htdocs， 运行 XAMPP (www.apachefriends.com) 的 Windows 用 户 ， 

口 /Applications/MAMP/htdocs, 运行 MAMP (www.mamp.info) 的 Mac 用 户 。 

如 果 你 不 知道 Web 文 档 根 目录 是 什么 ， 查 看 Web 服 务 器 应 用 程序 文档 或 操作 系统 文档 (如 果 
操作 系统 内 置 Web 服 务 应 用 程序 )。 

将 PHP 脚 本 放 到 正确 的 位 置 后 ， 就 可 以 用 浏览 器 执行 该 脚本 了 。 

沪 在 浏览 器 中 测试 脚本 












































(1) 打开 喜爱 的 Web 训 览 器 。 

大 多 数 情 况 下 ,， PHP 在 不 同 的 浏览 器 中 表现 相同 (因为 PHP 是 在 服务 器 上 运行 的 ), 因此 可 以 
随意 使 用 你 所 喜爱 的 浏览 器 。 在 本 书 中 ， 将 主要 使 用 Firefox 和 Safari 对 操作 系统 没有 任何 限制 。 

(2) 在 浏览 器 的 地 址 栏 中 ， 输 入 存放 脚本 的 网 站 URL，。 

在 本 书 的 示例 中 将 使 用 www.larryullman.com， 但 是 读者 的 URL 必 定 不 同 。 

如 果 在 本 机 运行 PHP， 那 么 URL 将 是 http:Wlocalhost (在 Windows 系 统 中 ) 或 http://localhost/ 
~username (在 Mac OS XX 中 )， 请 将 username 禁 换 为 所 使 用 的 真实 用 户 名 。 一些 多 合 一 程序 包 (如 
MAMP 和 XAMPP) 还 需要 在 URL 中 加 入 0 DUODOD 例如 : http:Wlocalhost:8888D0 

如 果 不 清楚 要 使 用 的 URL， 可 以 查看 Web 服 务 器 应 用 程序 文档 。 

(3) 向 URL 中 添加 /phpinfo.php。 

如 果 你 将 脚本 放 在 Web 文 档 根 目录 的 子 目录 下 ， 还 应 该 在 URL 中 加 上 子 目 录 名 (如 
/ch01/phpinfo.php), 

(4) 按 回 车 以 加 载 这 个 URL，。 

浏览 器 窗口 将 加 载 这 个 页 面 (参见 图 1-7)。 

如 果 看 到 PHP 代 码 (参见 图 1-8) 或 者 空白 页 ， 可 能 是 由 于 以 下 儿 种 情况 造成 的 : 

口 没有 正确 使 用 ULR 加 载 PHP 脚 本 〈 如 地 址 名 前 面 漏 掉 了 http ) ; 
口 服务 器 上 没有 局 用 PHP， 

口 没有 使 用 正确 的 扩展 名 。 

如 果 出 现 file not found 或 类 似 错误 (参见 图 1-9)， 可 能 是 由 于 : 
口 输入 的 URL 不 正确 ， 

口 PHP 脚 本 没 在 放 在 正确 的 目录 下 ，; 

口 PHP 脚 本 的 名 字 或 扩展 名 不 正确 。 
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System Darwin Lany-UllImans-iMac.local 10.4.1 Darwin Kernel Yersion 10.4.1: Fri Jul 16 23:04:20 
PDT 2010; rootixnuy-1504.7.51~1!RELEASE_I386 i386 

Build Date “|Mar 5 2010 16:41:09 

Configure “configure’ --with-mysql=!ApplicationsivtAMIP!L ibrary" ‘~-vvith- 

Command apxs2=!ApplicationsIMAMP!Library!bintapxs” ~-with-gd" -with-jpeg- 
dir=lApplicationsIMAMPILibrary’ -with-png-dir=!ApplicationsIMAMPILibrary’ -with-zlib ~- 
with-freetype-dir=|ApplicationsIMAMPILibrary’ -prefix=lApplicationsIMAMPIbiniphp5.3…- 
exec-prefix=lApplicationsIMAMPIbiniphpS.3 -sysconidir=|ApplicationsiIMAMPIconfiphpS.3" ~- 
With-soap” with-config-file-path=!ApplicationsIMAMPIconfiphps.3 -enable-back-Yars -- 
enable-bcmath -enable-hp ~-enable-gd-native-tf -with-bz2=iusr -with-ldap -with- 

1i= 愉 PPlicationsIiMAMPILibraryibinmysqlL_config -with-sqlite -with- 邮 -With- 
blib=lApPPlicationslMAMPILibrary”--enable- ‘mbstring=all With- 
curl=lApplicationsIMAMPILibrary' ‘--enable-dbx' -enable-Sockets -enable-bcmath th 
imap-=sharedJApplicationeIMAMPILiorarylliblimap-2007e -enable-soap” -Wihkerberos ~ 
enable-calendar’ -with-pgsql=sharec,ApplicationsIMAMPILioraryipg’ -enable-dbase” ~ 
enable-exif’ --YWith- rlibxml-dir=!ApplicationsiMAMPILibrary” ~ --With- 
getiext=shared,ApplicationsIMAMPILibrary -with-xsl=!ApplicationsivANPILibrary -with- 
pdo-mysql=shared,!ApplicationsiMANMP!Library’ -with-pdo- 
pgsql=shared,!ApplicationsiMAMPILibrarylpg’ ~-with- 
mcrypt=shared,!ApplicationsIMAMPILibrary’ *--with-openssl' 

Server AP| |Apache 2.0 Handler 

Yirtual disabled 

Directory 

Support 

Configuration |IApplicationsIMAMPIconfiphps.3 

File (php.ini) 

Loaded LibrarylApplication SupportappsoluteIMAMP PROrconfiphp.ini 

Configuration 

File 

Scanthis dr |(none) 

foradditional 

-inifiles 

Additional .ini|(none) 

files parsed 

PHP API 20090626 

PHP 20090626 

人 ITTA NS 
图 1-7 如 果 脚 本 执行 正确 ， 浏 览 器 将 会 如 图 所 示 
<?php 
phpinfo!( ); 
?> 
Y > i 了 2 二 人 
图 1-8 ”如 果 看 到 PHP 源 代码 ， 说 明 没有 执行 代码 








.OO 404. Not FOUnd | 
Not Found 


The requested URL /phpinf.php was not found on this server. 








图 1-9 ”服务 器 的 响应 说 明 服务 器 上 没有 找到 URL 访 问 的 文件 


V 提示 
口 不 能 像 用 其 他 应 用 程序 


训 打 开 HTML 页 面 或 其 他 文件 那样 直接 使 用 浏览 器 
点 非常 重要 ! PHP 脚 本 必须 使 用 Web 服 务 器 来 进行 处 理 ， 这 意味 着 必须 通过 一 个 URL 来 进 


行 访 问 〈 以 http:/ 开 头 的 地 址 )。 








打开 PHP 文 件 , 这 
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口 即使 不 是 一 个 资深 的 计算 机 专家 , 也 应 该 考虑 在 自己 的 计算 机 上 安装 PHP。 这 么 做 并 不 困 
难 ， 而 且 PHP 是 免费 的 。 同 样 ， 详 情 请 参考 附录 A。 

口 从 技术 上 说 , 并 不 需要 向 phpinfo () 脚本 中 添加 任何 HIML 代 码 。 如 果 不 添 加 , phpinfo () 
函数 将 会 生成 完整 的 HTML 页 面 。 


1.5 向 浏览 器 发 送 文 本 


如 果 只 能 知道 服务 器 是 不 是 支持 PHP，PHP 将 显得 不 那么 有 用 。 使 用 PHP 进 行 的 最 频繁 的 操 
作 是 以 纯 文本 和 HTML 标 签 的 方式 向 浏览 器 发 送信 息 。 比 如 使 用 print: 

print "something"; 

直接 输入 print， 后 面 添加 希望 显示 的 内 容 ， 例 如 一 个 简短 的 消息 、 变 量 的 值 、 计 算 的 结果 
等 。 上 述 函 数 中 的 参数 是 要 打印 文本 。 因 为 这 个 参数 是 文本 字符 串 ， 所 以 必须 将 它们 用 引号 引起 
来 (相对 而 言 ， 数 字 则 不 需要 使 用 引号 )。 

需要 说 明 的 是 ，print 并 不 会 真正 打印 出 什么 东西 ， 它 只 是 输出 数据 。 当 使 用 Web 浏 览 器 运 
行 PHP 脚 本 时 ，PHP 输 出 的 数据 只 会 显示 在 浏览 器 上 。 

也 请 注意 在 行 结 尾 处 有 一 个 分 号 (;)。 每 个 PHP 代 码 中 的 语句 必须 使 用 分 号 作为 结尾 ， 忘 记 
这 个 必要 条 件 是 发 生 错误 的 常见 原因 。PHP 中 的 语句 是 一 行 可 执行 的 代码 ， 例 如 


print "something"; 






























































或 

phpinfo(); 

相反 ,在 本 书后 面 讨论 的 行 注释 、PHP 标 签 、 控 制 结构 (条件 、 循 环 等 ) 以 及 某 些 其 他 结构 
都 不 应 该 使 用 分 号 。 


最 后 ， 你 应 该 了 解 一 些 辅助 技术 细节 : Print 不 是 真正 的 国 数 ， 它 是 一 种 DODOD ， 而 
phpinfo () 是 真正 的 国 数 。 虽 然 把 print 当 作 国 数 也 是 合理 的 ， 但 print 是 一 种 语言 结构 ， 在 使 
用 时 不 需要 加 括号 ， 如 上 面 的 例子 所 示 。 








这 打印 一 条 简单 的 消息 











(1) 在 文本 编辑 器 或 者 IDE 中 新 建 一 个 HTMLIL 文 档 ， 命 名 为 hello .php (参看 脚本 1-3): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

> "http://www.w3.0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Hello, World!</title> 

</head> 

<body> 

<p>The following was created by PHP: 


上 述 代 码 基 本 上 全 是 标准 HTML。 最 后 一 行 用 来 区 别 硬 编码 的 HTML 和 由 PHP 生 成 的 HTML。 
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脚本 1-3 通过 在 PHP 标 签 间 放置 print 语 句 ， 服 务 器 将 动态 地 向 浏览 器 发 送 Hello, world! 问 候 


1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
嫩 <html Xmlns="http://www.w3.org/1999/xXhtml" 

xml:lang="en" lang="en"> 

<head> 
5 <meta http-equiv="content-type" 

content="text/html; charset=utf-8" /> 

6 <title>Hello, World!</title> 
7 </head> 
8 <body> 
9 <p>The following was created by PHP: 
10 <?php 
11 print "Hello, world!"; 
12. .33 
13 </p> 
14 </body> 
15 </html> 


(C) 在 下 面 新 的 一 行 ， 输 入 <?php 来 创建 PHP 起 始 标签 。 

(3) 添加 : 

print "Hello, world!"; 

打印 短语 Hello, world! 是 大 多 数 编程 参考 书目 所 教授 的 第 一 步 。 尽 管 这 对 于 使 用 PHP 来 说 是 
微不足道 的 理由 ， 但 是 如 果 还 没有 编写 过 至 少 一 个 Hello, world! 应 用 程序 的 话 ， 你 将 不 能 称 之 为 
一 个 真正 的 程序 员 。 

(4) 关闭 PHP 部 分 以 及 HIML 页面 : 


2 

和 > 
</body> 
</html> 


(5) 将 文件 保存 为 hello .php， 放 在 启用 了 PHP 的 服务 器 上 ， 然 后 在 浏览 器 中 测试 (参见 图 
1-10)。 














MD 











AMAO Hello, World! 


The following was created by PHP: Hello, world! 























图 1-10 个 简单 的 Hello, world! 示 例 : 第 一 个 PHP 编 程 尝试 





























如 果 在 自己 的 计算 机 上 运行 PHP， 请 记 住 将 文件 保存 在 适当 的 目录 下 ， 并 且 通 过 访问 
http://localhost/ 来 访问 脚本 。 

如 果 你 看 到 错误 或 者 空白 页 而 不 是 如 图 所 示 的 正确 结果 ， 请 查看 本 章 最 后 的 调试 部 分 。 

ww 提示 

口 PHP 是 不 区 分 大 小 写 的 ， 因 此 当 通 过 print、Print 和 PRINT 调 用 print 或 phpinfo() 国 
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数 时 结果 都 是 一 样 的。 在 本 书后 面 的 部 分 中 (如 第 2 章 ) 将 看 到 大 小 写 起 重要 作用 的 示例 。 
口 可 以 使 用 其 他 函数 (例如 ，echo 和 printf() ) 向 浏览 器 发 送 文 本 , 但 是 在 本 书 中 ,将 主 
要 使 用 print。 

口 通常 情况 下 可 以 使 用 print 函 数 作用 于 多 行文 本 : 
print "This is a longer sentence of text."; 


以 右 引 号 〈") 结束 要 打印 的 消息 ， 并 在 语句 的 末尾 加 上 分 号 (; )。 


1.6 使 用 PHP 手册 


可 以 在 线 访问 PHP 和 手册 (www.php.net/manual) ， 它 列举 了 在 这 种 语言 中 可 以 使 用 的 每 一 个 函 
数 ， 但 是 使 用 手册 需要 一 些 专业 知识 。 该 手册 按照 使 用 PHP 的 先后 顺序 进行 组 织 (安装 、 语 法 、 
变量 )， 最 后 是 专题 函数 (MySQL、 字 符 串 函数 等 ) 部 分 。 

要 迅速 地 在 PHP 手 册 中 找到 函数 ， 请 在 Web 浏 览 器 中 访问 www.php.net/functionname (例如 ， 
www.php.net/print ) 。 

为 了 理解 如 何 描述 函数 ， 请 查看 print 国 数 页 的 开头 〈 参 见 图 1-11) : 
































print 
(PHP 4, PHP 5) 


print 一 Output a string 





= Description Report a bug 














int print ( string $arg ) 





Outputs arg. 


print() is not actually a real function (it is a language construct) so you are not required to use 
parentheses with its argument list. 











图 1-11 ”PHP 和 手册 中 的 print 函 数 结构 


第 一 行 是 函数 本 身 的 名 称 ， 然 后 是 可 以 使 用 这 个 函数 的 PHP 版 本 。 由 于 语言 在 发 展 ,会 不 断 
加 入 的 新 函数 ， 也 会 偶尔 移 除 旧 函数 。 接 下 来 是 对 这 个 函数 的 文字 描述 以 及 函数 的 基本 用 法 。 用 
法 是 最 重要 也 是 最 容易 混淆 的 部 分 。 
在 这 个 示例 中 ， 第 一 个 值 int 表 明 print 返 回 一 个 整 型 值 (print 正 常情 况 返 回 1)。 在 圆 括 
号 内 string S$arg 说 明 该 函数 必须 接受 一 个 参数 ， 而 且 必 须 是 字符 串 形 式 。 我 们 已 经 实际 使 用 
过 了 。 

作为 对 比 ， 在 手册 中 找到 n12br () (参见 图 1-12)。 这 个 函数 将 文本 中 的 换行 符 (相当 于 按 
下 Return 或 Enter 键 ) 替换 成 HTML 的 换行 标签 。 这 个 函数 返回 一 个 字符 串 ， 将 一 个 字符 串 作为 第 
一 个 参数 , 将 一 个 可 选 的 布尔 型 (TRUE/FALSE) 作为 第 二 个 参数 。 当 你 在 手册 中 看 到 方 括号 时 ， 
就 表示 其 内 的 参数 是 可 选 参 数 ， 这 些 可 选 参数 会 列 在 最 后 。 如 果 函 数 接受 多 个 参数 ， 要 使 用 过 号 
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将 它们 分 隔 开 来 。 你 可 以 这 样 调用 该 函数 : 


nl2br("Some text"); 
nl2br("Some text", false); 


国 数 的 定义 还 说 明 第 二 个 参数 有 一 个 默认 值 IRUE ， 表 示 如 果 不 给 第 二 个 参数 传人 
是 将 默认 创建 XHTML 的 <br/> 标 签 。 在 上 例 中 ,该 函数 会 创建 HTML 的 <br> 标 签 


nl2br 
(PHP 4, PHP 5) 














nl2br 一 Inserts HTML line breaks before all newlines in a string 








= Description Report a bug 


string nl2br ( string $string [, bool $is_xhtml = true ] ) 


Returns string with '<br />' or '<br>' inserted before all newlines. 























图 1-12 ”PHP 手册 中 的 nl12br () 函数 


Le 雪 而 迷恋 或 者 不 知道 如 何 正确 使 用 它 ,请 查看 PHP 手 册 参 考 页 以 寻求 帮助。 


(1) 在 你 的 Web 浏 览 器 中 输入 www.php.neUjiunctiozmzzzamaze 
如 果 PHP 手 册 中 没有 与 你 输入 的 函数 匹配 的 记录 ,查看 国 数 拼 写 是 否 正确 或 看 看 手册 中 推荐 
的 替代 函数 (参见 图 1-13) 。 


PHP Function List 














Sorry, but the function n2br is not in the online manual. Perhaps you misspelled it, or it is a relatively new function that hasn't made it 
into the online documentation yet. The following are the 20 functions which seem to be closest in spelling to n2br (really good matches 
are in bold). Perhaps you were looking for one of these: 























nl2br bcsqrt long2ip 
png2wbmp finfo_buffer bzerrno 
atan2 iconv substr natsort 
mongodbref bzerror opendir 
substr bin2hex strpbrk 
hebrev deg2rad mongodb 
bzread scandir 


If you want to search the entire PHP website for the string "n2br", then click here. 








For a quick overview over all documented PHP functions, click here. 











图 1-13 ”如 果 URL 中 输入 的 函数 名 不 完全 匹配 ，PHP 和 手册 会 列 出 一 些 替代 函 


(2) 将 函数 所 在 的 PHP 版 本 与 你 使 用 的 PHP 版 本 作 比 较 。 

使 用 前 面 介绍 过 的 phpinfo() 函数 , 确认 你 使 用 的 PHP 版 本 。 如 果 某 个 函数 是 在 后 面 的 版 本 
中 加 入 的 ， 你 可 以 升级 PHP 版 本 ， 也 可 以 找 其 他 方法 实现 。 

(3) 检查 函数 返回 的 数据 类 型 。 

有 时 会 因为 函数 返回 值 而 出 现 问 题 ， 比 如 它 返 回 的 类 型 并 不 是 你 想 要 的 。 

(4) 检查 函数 需要 (或 接受 ) 的 参数 数量 和 参数 类 型 。 
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使 用 函数 的 常见 错误 是 ， 在 调用 函数 时 传 入 错误 的 参数 数量 和 参数 类 型 。 

(5) 如 果 有 用 户 注释 ， 请 多 加 关注 ， 你 会 从 中 学 到 更 多 知识 

用 户 注释 有 时 非常 有 用 (有 时 没什么 用 )。 

Vv 提示 

口 如 果 你 看 到 一 条 警告 消息 ， 指 出 某 个 函数 已 经 过 时 (参见 图 1-14)， 就 表示 这 个 函数 会 在 
未 来 的 PHP 版 本 中 移 除 ， 你 应 该 使 用 更 新 的 、 更 好 的 禁 代 函 数 ( 可 以 确定 , 一 定 会 有 一 个 

函数 替代 它 )。 


ereg 





























(PHP 4, PHP 5) 


ereg 一 Regular expression match 





-| Description Report a bug 

















int ereg ( string gpattern , string $string [, array &gregs ] ) 


Searches a string for matches to the regular expression given in pattern in a case-sensitive way. 


Warning 
This function has been DEPRECATED as of PHP 5.3.0. Relying on this 
feature is highly discouraged. 


图 1-14 ”在 代码 中 应 该 避免 使 用 过 时 的 函数 


























1.7 向 浏览 器 发 送 HTML 


首次 学 习 HTML 的 人 都 会 很 快 发 现 ， 在 Web 浏 览 器 中 查看 纯 文本 还 有 很 多 不 足 之 处 。 确 实 ， 
创造 HTML 的 初 豆 就 是 为 了 让 纯 文本 更 加 吸引 人 而 且 有 用 。 由 于 HTML 通 过 向 文本 添加 标签 来 工 
作 ， 因 此 可 以 使 用 PHP 向 浏览 器 发 送 HTML 标 签 以 及 其 他 数据 : 


print "<b>Hello, world!</b>"; 


这 里 有 一 种 情况 需要 留意 ,对 于 需要 使 用 引号 的 HTML 标 签 (如 <a href="page.php">1ink 
</a>) 将 在 使 用 PHP 进 行 打印 时 发 生 问题 ， 因 为 print 国 数 也 使 用 引号 (参见 图 1-15): 


print "<a href="page.php">1ink</a>"; 

解决 方法 是 通过 在 引号 前 面 加 上 反 斜 线 ( 作 ) 的 方式 将 其 转 义 : 

print "<a href=\"page.php\">1ink</a>"; 

通过 将 print 语 句 中 的 引号 进行 转 义 ，PHP 会 打印 字符 本 身 ， 而 不 是 将 其 解释 为 需要 打印 的 
字符 串 的 起 始 和 结尾 
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Parse error: syntax error, Unexpected T_STRING in 
/Users/larryullman/Sites/phpvqs4/hello.php on line 11 














图 1-15 ”尝试 打印 双 引 号 会 出 现 错误 提示 ， 因 为 print 语 句 中 也 要 使 用 双 引 号 





> 向 浏览 器 发 送 HTML 的 步 台 


(1) 在 文本 编辑 器 或 IDE 中 打开 hello .php 脚 本 (参看 脚本 1-3 ) 。 
(2) 编辑 第 11 行 中 Hello, world! 文 本 ,添加 HTML 标 签 ， 如 下 所 示 (参看 脚本 1-4): 


print "<span style=\"font-weight: bold;\">Hello, world!</span>"; 








脚本 1-4 使 用 print 国 数 ， 可 以 将 HTML 标 签 和 文本 一 起 发 往 浏 览 器 ， 浏 览 器 会 根据 标签 对 文 
本 进行 格式 化 








< <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

急 <html Xmlns="http://www.w3.org/1999/xhtm1" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>Hello, World!</title> 

7 </head> 

8 <body> 

9 <p>The following was created by PHP: 

10 <?php 

11 print "<span style=\"font-weight: bold;\">Hello, world!</span>"; 

12 .2 

13 </p> 

14 </body> 


15 </html> 
让 消息 中 PHP 生 成 的 部 分 突出 显示 ， 我 们 应 用 了 一 些 CSS， 用 粗 体 显示 它们 。 要 达到 这 个 目 
需要 在 span 标 签 中 使 用 反 斜 线 来 转 义 引号 ， 以 避免 和 print 语 句 中 的 引号 发 生 冲 突 。 

(3) 将 脚本 保存 为 hel1lo2 .php, 放置 在 启用 了 PHP 的 服务 器 上 ,并 且 在 浏览 器 上 运行 该 页 ( 参 
1-16)。 





于 





汉 
现 








AMA Hello, World! 


The following was created by PHP: Hello, world! 














图 1-16 ”页 面 的 新 版 本 有 了 一 定 的 装饰 和 吸引 力 


(4) 查看 HTML 页 的 源 代码 以 查看 发 送 到 浏览 器 的 代码 (参见 图 1-17)。 
怎样 做 将 取决 于 所 使 用 的 浏览 器 : 在 Firefox 中 选择 View>Page Source, 在 正中 为 View>Source， 
在 Safari 中 是 View>View Source。 
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AMO Source of http://phpvqs4:8888/hello2.php 


<IDOCTYPE html PUBLIC "-# AW3CA72DTD XHTML 1.8 Transitionagl//EN" 
"http:/ Awww .w3.0rg/TRA/xhtml1zDTDA/xhtml1-transitional .dtd"> 
tml xmlns="http:A /Www .w3.0rg/19997xhtml" xml:lang="en" lang="en"> 


eqd> 
eta http-equiv="content-type" content="text/html; charset=utf-8" /> 
titlesHello, World!l</tit le 

</head> 

击 ody> 

中 >The following was created by PHP: 

<span style="font-weight: bold;">Hello, world!l</span>—/ p> 

<*body> 

<*html> 




















图 1-17 Hel102 .php 页 中 HTML 源 文件 的 显示 结果 (参看 脚本 1-4) 

有 必要 将 这 一 步 作 为 习惯 操作 ， 尤 其 是 当 有 错误 发 生 时 。 请 记 住 PHP 主 要 用 来 生成 HTML， 
然后 向 Web 浏 览 器 发 送 并 通过 浏览 器 解释 它们 。 通 常 确认 向 Web 浏 览 器 发 送 的 内 容 (用 查看 源 文 
件 的 方式 ) 将 有 助 于 发 现在 浏览 器 解释 过 程 中 出 现 的 问题 。 

w 提示 
口 对 于 PHP 程 序 员 来 说 , 理解 引号 的 作用 以 及 如 何 对 有 问题 的 字符 进行 转 义 至 关 重 要 。 接 下 
来 的 两 章 将 更 详细 地 介绍 这 些 主题 。 

口 通过 PHP 向 Web 浏 览 器 发 送 的 HTML 不 必 如 此 简单 。 还 可 以 创建 表格 、JavaScript 以 及 更 多 
口 请 记 住 ， 任 何在 PHP 标 签 之 外 的 HTML 都 将 自动 发 送 至 浏览 器 。 在 PHP 标 签 中 ，print 语 
句 被 用 来 向 浏览 器 发 送 HTML。 









































使 用 空白 

加 
DO00000000000000000000PHPIUUUUUUUUNUPHR UU Web 
D00000000000000000D000PHFPIOU0O0UUOUUUIUUUUUOO 
UUOUU0O0OU00D0 

U00UwebUUHIMLUOUOUOUUOUUOOUUOOOOOOUOOUOO0O0O0O0O0 
UOUUUOUU0OO0O0O0OO0O0O0O0O0U0O0OU0O0OO0OUOO0O0OUUHIMLIUOUUODODO 
UUO0U0O0OU000000HIMIUOOUOUUOUUOOOO0O0OO0ORPHEUUOOUUOODO 
UUOUUUOUUOOU0O0O0OO0OO0O0O0OU0O0OUxSUOHIMLUUOODODO 


Toe i ale Slav IN a 


UUUU0O0O000HIMIOOUOUOUOUOUOUOUOOUOOOOO0O0O0O0D 
D000000000000000UUUCSSUUD <p>U <aiv>U0U00000 
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1.8 为 脚本 添加 注释 "| 


注释 是 对 编程 的 完善 , 这 并 不 是 因为 它 可 以 做 什么 , 而 是 因为 它 能 够 帮助 程序 员 今后 想起 为 
什么 这 么 做 。 计 算 机 在 处 理 脚本 的 时 候 将 忽略 这 些 注释 。 此 外 ，PHP 注 释 将 不 会 被 发 送 至 Web 浏 
览 器 ， 因 此 能 够 保守 秘密 。 

PHP 提 供 3 种 添加 注释 的 方式 。 可 以 在 希望 不 运行 的 单行 代码 开头 使 用 / /或 者 # 将 其 注释 掉 。 
例如 : 

// 这 是 一 条 注释 。 

还 可 以 使 用 / /或 者 # 在 PHP 语 句 的 结尾 添加 注释 ， 例 如 : 

print "Hello"; // 只 是 一 名 问候 。 

虽然 是 一 个 使 用 习惯 问题 ， 但 在 PHP 中 /7/ 比 # 更 常用 。 

还 可 以 在 开头 和 结尾 使 用 /* 和 */ 注 释 多 行 : 














/* 这 是 
多 行 注释 。 
a 喜爱 这 种 注释 风格 ， 因 为 它 包括 开始 和 结束 “标签 ”， 为 广 释 划分 了 明显 的 开始 


(1) 在 文本 编辑 器 或 IDE 中 打开 he11o2 .php (参看 脚本 1-4)。 
(2) 在 PHP 起 始 标签 后 为 脚本 添加 一 些 广 释 (参看 脚本 1-5): 








* ”文件 名 : hello3 .php 
* ” 书 中 引用 : 脚本 1-5 
* ”由 Larry Ullman 创 建 





脚本 1-5 在 一 行 代码 前 放置 / /或 #， 将 导致 PEHP 不 再 对 该 行进 行 处 理 





1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 
5 

6 

7 

8 








<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Hello, World!</title> 
</head> 
<body> 
9 <p>The following was created by PHP: <br/> 
10 <?php 
11 /* 
12  * 文 件 名 : hello3.php 
13 * 书 中 引用 : 脚本 1-5 
14 * 由 Larry Ullman 创 建 
15 6 
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//print "<span style=\"font-weight: bold;\">Hello, world!</span>\n"; 


?> 

<!-- 这 是 一 个 HTML 注 释 。--> 
</p> 

</body> 

</html> 


这 只 是 能 编写 的 多 种 注释 中 的 一 个 示例 。 本 书 强 烈 建议 读者 使 用 注释 记录 下 脚本 的 功能 , 它 
依赖 什么 信息 ， 由 谁 创 建 ， 什 么 时 候 创 建 等 信息 。 看 样式 就 知道 ， 这 种 注释 一 般 都 放置 在 脚本 的 
顶部 〈 作 为 PHP 代 码 段 的 开头 )。 额 外 的 星 号 并 不 是 必需 的 ， 它 们 只 是 强调 这 是 一 个 注释 。 

(3) 在 第 17 行 的 print 语 句 之 前 输入 //。 

在 print 语 句 之 前 添加 // 可 以 用 来 将 这 个 函数 注释 掉 ， 这 就 意味 着 它 将 不 会 被 执行 。 

(4) 在 关闭 PHP 标 签 之 后 (第 19 行 )， 添 加 一 个 HTML 注 释 : 


<!-- 











这 是 一 个 HTML 注 释 。 


这 行 代码 将 有 助 于 你 理解 不 同 的 注释 以 及 它们 在 哪里 出 现 。 这 个 注释 将 只 在 HTML 源 代码 中 
出 现 。 
(5) 将 脚本 保存 为 hne11o3 .php， 放 置 在 启用 了 PHP 的 服务 器 上 ， 然 后 在 Web 浏 览 器 上 运行 该 




















页 (参见 图 1-18)。 
AMOA Hello, World! 
The following was created by PHP: 
图 1-18 ” 当 print 语 句 被 注释 掉 后 ， 页 面 看 起 来 如 同 print 函 数 根本 没有 出 现在 代码 中 一 样 

















(6) 查看 该 页 的 源 代码 以 了 解 HTML 注 释 (参见 图 1-19)。 





AMA Source of http://phpvqs4:8888/hello3.php 


<IDOCTYPE html PUBLIC "~-#AAW3CA ADTD XHTML 1.8 Transitiongl//EN" 
"http:sA www .Ww3.0rgA/TRAxhtml1lzDTDAxhtml1i-transitional .dtd"-> 

tml xmlns="http: A /www .Ww3 .0rg/19997xhtml" xml:lang="en" lang="en"»> 

eqd> 
eta http-equiv="content-type" content="textihtml; charset=utf-8" /> 
titlesHel lo, World!l</tit le 

<*head> 

body> 

< 中 >The following was created by PHP: -出 TY /> 

=<!-- This is an HTML comment. ——> 

<p> 

<body> 

<ehtmil> 











图 1-19 HTML 注 释 不 会 出 现在 Web 浏 览 器 中 ,但 是 会 存在 于 HTML 源 代码 里 。PHP 的 
注释 将 只 出 现在 服务 器 上 的 PHP 脚 本 中 
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Vv 提示 

口 可 以 使 用 /* 和 */ 将 单行 代码 或 者 多 行 代码 注释 掉 。/ /或 者 # 一 次 只 能 注释 掉 一 行 代码 。 

口 不 同 的 程序 员 偏 爱 不 同 的 代码 注释 方式 。 最 重要 的 是 找到 一 种 对 自己 适用 的 方式 ， 并 且 

坚持 使 用 。 

D 请 注意 , 不 能 在 PHP 代 码 中 使 用 HTML 注 释 符号 (<!-- 和 -->) 来 注释 代码 。 可 以 使 用 PHP 
向 浏览 器 中 打印 这 些 标 签 ， 但 是 在 这 种 情况 下 ， 你 就 已 经 创建 一 个 注释 ， 因 为 它 出 现在 
客户 端 计算 机 上 的 HTML 源 代码 中 (而 不 是 浏览 器 窗口 中 )。 而 PHP 注 释 永远 不 会 出 现在 
用 户 的 计算 机 上 。 

口 尽管 我 十 分 坚信 你 不 会 过 度 注释 代码 ， 但 是 在 本 书 中 将 不 会 注释 所 有 脚本 ， 这 是 出 于 
节省 纸 面 空间 的 考虑 。 但 是 本 书 将 对 脚本 的 名 称 及 编号 给 出 注释 ， 以 达到 对 照 参考 的 
目的 。 

口 当 你 修改 脚本 代码 时 ， 不 要 忘记 同时 更 新 注释 。 如 果 注 释 中 的 说 明 与 脚本 、 代 码 行 的 行 
为 相悖 ,会 让 人 困惑 。 

















1.9 调试 的 基本 步骤 


调试 不 是 一 个 可 以 简单 掌握 的 概念 ， 它 是 通过 实际 动手 才能 真正 掌握 的 一 门 技术 。 即 便 是 接 
下 来 的 50 页 专门 针对 这 个 主题 进行 阐述 ， 候 怕 你 也 只 能 掌握 必 备 调试 技能 的 一 小 部 分 。 

本 书 在 这 里 如 此 介绍 调试 的 原因 是 想 让 你 记 住 , 在 想 清 楚 之 前 不 要 匆忙 地 去 编程 。 有 时 代码 
并 没有 像 期 望 的 那样 工作 ， 而 程序 员 不 可 避免 地 会 造成 政 漏 和 错误 ， 这 会 令 人 痛苦 并 抓 狂 ， 即 
便 是 使 用 像 PHP 这 样 友好 的 语言 也 是 如 此 。 我 从 1999 年 开始 使 用 PHP 编 程 ， 但 是 偶尔 还 是 会 陷入 
编程 的 泥沼 中 , 因此 调试 是 需要 掌握 的 重要 技能 ,读者 将 最 终 认识 到 它 的 必要 性 并 深 诬 体会 到 这 
一 点 。 作 为 PHP 编 程 历险 的 开始 ， 本 书 将 提供 下 面 这 些 基 础 但 又 具体 的 调试 技巧 。 


> 调试 PHP 脚 本 




















口 确认 总 是 通过 URL 来 运行 PHP 脚 本 ! 这 可 能 是 初学 者 常 犯 的 错误 。PHP 代 码 必 须 通 过 
Web 服 务 器 应 用 程序 来 运行 ， 也 就 是 说 必须 通过 http://something 发 出 请 求 。 如 果 你 看 到 
的 是 PHP 代 码 ， 而 不 是 运行 结果 的 话 ， 就 说 明 PHP 脚 本 并 没有 通过 URL 执 行 。 

口 熟知 运行 的 PHP 版 本 。 一 些 问 题 是 由 PHP 的 版 本 所 致 。 在 使 用 启用 PHP 的 服务 器 之 前 ， 运 

行 phpinfo.php 文 件 (参看 脚本 1-2) 来 确定 使 用 的 PHP 版 本 。 

口 确保 display_errors 是 开启 状态 。 这 是 一 个 PHP 基 础 配置 设置 (在 附录 A 中 有 相关 的 讨 
论 ) 。 要 确认 这 些 配 置 ， 可 以 执行 bhpinfo() 函数 (使 用 浏览 器 ， 在 结果 页 中 搜索 
display_errors)。 出 于 安全 的 考虑 ，PHP 可 能 没有 设置 显示 所 发 生 的 错误 。 如 果 是 这 
样 ， 将 在 问题 发 生 时 只 能 看 到 空白 的 页 面 。 为 了 对 该 问题 进行 调试 ， 需 要 查看 这 些 错误 ， 
因此 在 学 习 的 时 候 请 将 这 个 设置 打开 。 可 以 在 附录 A 和 第 3 章 中 找到 相关 的 介绍 。 


















































22 010 PHPDOD 








口 检查 HTML 源 代码 。 有 了 时 间 题 隐藏 在 页 面 的 HTML 源 代码 中 。 事 实 上 ， 有 时 PHP 错 误 消 息 

正 是 隐藏 在 那里 的 ! 

口 相信 错误 消息 ! 初学 者 常 犯 的 另外 一 个 错误 是 不 完整 阅读 PHP 报 告 的 错误 消息 , 或 者 不 完 
全 相信 这 些 错误 。 尽 管 这 些 错误 消息 有 了 时 非常 含糊 ， 并 且 看 上 去 没有 什么 意义 ,但 是 还 
是 不 能 忽略 它 ! 至 少 在 错误 发 生 于 哪 行 代码 这 个 问题 上 ，PHP 通 常 都 是 正确 的 。 如 果 需 
要 将 错误 信息 转发 给 别人 〈 比 如 你 请 我 帮 你 分 析出 错 原因 时 ) ， 一 定 广 意 要 包括 全 部 错 
误 信息 。 

口 休息 一 下 ! 这 么 多 年 来 我 遇 到 过 很 多 的 编程 问题 ， 并 且 绝 大 多 数 最 困难 的 问题 ， 都 是 在 
我 离开 计算 机 去 散步 一 会 回来 后 解决 的 。 编 程 时 很 容易 陷入 挫败 与 迷惑 交织 的 痛苦 中 ， 
在 这 种 情况 下 ， 所 采取 的 任何 进一步 措施 似乎 都 会 将 事情 弄 得 更 精 。 

w 提示 

口 这 是 一 些 常 见 的 调试 技巧 , 适用 于 PHP 编 程 初学 者 。 对 于 现在 来 说 应 该 已 经 足够 ， 因为 本 

书 中 的 相关 示例 都 非常 简单 。 代 码 越 复杂 ， 对 调试 技术 的 要 求 也 就 越 高 ， 因 此 我 编著 的 

《PHP 6 与 MySQL 5 基础 教程 》 专 门 用 一 章 来 讨论 这 个 主题 。 


1.10 回顾 和 实践 


本 书 会 在 每 章 的 最 后 添加 “回顾 和 实践 ”一 节 。 在 本 节 中 ， 我 会 就 本 章 涉 及 的 内 容 提出 一 些 
问题 ， 目 的 是 巩固 和 扩展 你 学 到 的 知识 和 技能 。 无 论 你 是 对 本 书 内 容 存 有 疑问 ， 还 是 在 自学 中 
遇 到 问题 ， 都 可 以 把 问题 发 到 本 书 的 论坛 上 (www.LarryUllman.comy/forum/) ， 我 们 会 为 你 答疑 
解 惑 。 
































1.10.1 回顾 


D 口 HIML 是 什么 ? XHTML 是 什么 ?这 两 者 之 间 有 什么 不 同 之 处 ? 

口 你 的 文本 编程 器 或 IDE 使 用 的 是 哪 种 编码 ? 它 与 你 在 HTML 代 码 中 指定 的 相同 吗 ? 
DCSS 是 什么 ， 用 来 做 什么 ? 

口 你 服务 器 上 PHP 脚 本 的 扩展 名 是 什么 ? 

口 什么 是 “Web 根 目录 ”? 你 的 服务 器 的 Web 根 目录 是 什么 ? 

口 如 何 测试 PHP 脚 本 ? 如 果 PHP 脚 本 没有 运行 ， 是 什么 原因 ? 

D 说 出 两 个 在 PHP 代 码 中 添加 注释 的 方法 。 说 说 注释 对 你 都 有 哪些 用 处 。 


1.10.2 ”实践 


口 如 果 需 要 访问 多 个 服务 器 ， 确 认 其 他 服务 器 上 运行 的 PHP 版 本 。 
口 创建 一 个 静态 HTML 网 页 ， 显 示 一 些 信息 。 用 PHP 生 成 一 些 内 容 替 换 网 页 上 部 分 静态 信 
自 
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口 创建 一 个 模板 ,该 模板 必须 包含 HTML 的 基本 框架 、 PHP 开 始 标签 和 结束 标签 以 及 一 些 基 | 本 
本 注释 信息 。 

口 使 用 phpinfo () 函数 确认 服务 器 上 已 启用 display_errors。 如 果 未 启用 ， 更改 服务 器 

配置 来 启用 它 (详细 内 容 见 第 3 章 和 附录 A)。 

口 后 面 章节 会 用 到 一 些 新 的 函数 ， 你 应 该 随时 查看 PHP 手 册 。 








wl 








本 章 内 容 

口 什么 是 变量 
口 变量 语法 
口 变量 类 型 
口 为 变量 赋值 
口 理解 引号 
口 回顾 和 实践 





在 第 1 章 中 , 已 经 使 用 PHP 向 Web 浏 览 器 发 送 了 简单 的 文本 和 HTML 代 码 一 一 换 句 话说， 做 这 
些 事 其 实 根 本 用 不 着 PHP! 别 担心 , 本 书 将 教授 你 如 何 联合 使 用 print 和 其 他 的 PHP 特 性 来 为 Web 
网 站 做 有 用 的 事 。 

为 了 实现 从 创建 简单 的 静态 页 面向 动态 Web 应 用 程序 和 交互 式 Web 网 站 的 飞跃 ， 需 要 使 用 变 
量 。 同 在 其 他 编程 语言 中 一 样 ， 变 量 也 是 PHP 中 的 一 个 核心 概念 。 理 解 什 么 是 变量 ， 一 门 语言 支 
持 什么 类 型 的 变量 ， 以 及 如 何 使 用 它们 ， 对 你 的 工作 至 关 重 要 。 

本 章 将 讨论 在 PHP 中 所 使 用 的 变量 的 基础 知识 ， 后 续 章 节 则 会 详细 介绍 各 种 变量 。 如 果 从 
未 接触 过 变量 ， 本 章 对 读者 来 说 将 是 很 好 的 入 门 。 如 果 已 经 深 详 此 道 ， 在 本 章 的 学 习 中 读者 将 
轻车熟路 。 


2.1 什么 是 变量 


最 好 的 理解 是 把 变量 想象 为 存储 数据 的 容器 。 一 旦 数据 被 存储 在 一 个 变量 中 (更 加 精确 地 
说 ,一 旦 变量 被 赋值 )， 数 据 /变量 将 能 够 被 改变 ， 在 Web 浏 览 器 上 打印 出 来 ， 存 储 在 数据 库 中 、 
被 电子 邮件 发 送 ， 等 等 。 

PHP 中 的 变量 本 质 上 是 灵活 多 变 的 : 可 以 为 一 个 变量 赋值 、 在 变量 中 检索 数据 (而 不 影响 变 
量 的 值 )、 为 变量 赋 新 的 值 ， 以 及 在 需要 的 时 候 进 行 循环 。 但 是 PHP 中 的 变量 也 是 临时 的 ， 它们 
只 存在 于 一 个 脚本 的 执行 期 间 。 当 脚本 的 最 后 一 个 PHP 标 签 被 执行 过 后 , 这 些 变量 也 就 不 存在 了 。 
当 用 户 单 击 一 个 链接 或 者 提交 表单 时 ， 将 获得 一 个 新 的 页 面 ， 同 时 这 些 变量 也 将 不 复 存在 。 
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在 深入 探讨 变量 之 前 ， 我 们 写 一 个 简单 的 脚本 ， 用 来 列 出 一 些 PHP 中 的 预定 义 变量 。 这 些 变 
量 不 需 创建 即 可 使 用 ， 这 是 因为 PHP 已 经 为 用 户 完成 了 创建 工作 。 在 本 书 的 课程 中 ,会 介绍 许多 
不 同 的 预定 义 变量 。 在 这 个 示例 中 ， 使 用 了 $_SERVER 这 个 预定 义 变量 。 它 包含 了 关于 运行 PHP 


的 计算 机 的 许多 信息 2 


print_z() 国 数 提供 了 一 种 显示 变量 值 的 简单 方式 : 

print_r($variable name); 

只 要 将 要 查看 的 变量 名 作为 print_r () 的 参数 就 可 以 显示 变量 的 值 (在 这 一 章 中 你 会 学 到 很 
多 关于 变量 的 语法 )。 


(1) 在 文本 编辑 器 或 者 IDE 中 创建 新 的 PHP 脚 本 ， 命 名 为 predefined.php (参看 脚本 2-1) 。 
脚本 2-1 调用 print < () 函数 以 便 查 看 储存 在 $_SERVER 变 量 中 的 值 


1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
号 <html Xmlns="http://www.w3.org/1999/xhtml" 

xml:lang="en" lang="en"> 
































4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Predefined Variables</title> 
7 </head> 

8 <body> 

9 <pre> 

10 <?php // 脚本 2-1 - predefined.php 

11 

12 // 显示 $_SERVER 变量 的 值 : 

13 print r ($ SERVER); 

14 

J 

16 </pre> 

17 </body> 

18 </html> 

(2) 创建 HTML 初 始 标 签 : 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Predefined Variables</title> 

</head> 

<body> 

<pre> 


这 段 代 码 复 用 了 在 第 一 章 中 创建 的 XHTML 模 板 。 在 页 面 的 body 部 分 ， 本 书 使 用 <pre> 标 签 
让 生成 的 PHP 信 息 更 加 清晰 易 读 。 尽 管 不 赞成 使 用 这 些 标签 二 它们 )， 
但 是 它们 能 够 很 好 地 达到 这 个 目的 。 如 果 不 使 用 <pre> 标 签 , 由 print_r() 函数 生成 的 结果 将 显 
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得 非常 凌乱 。 

(3) 添加 PHP 代 码 : 

<?php // 脚本 2-1 - predefined.php print_r($_SERVER); ?> 

这 段 PHP 代 码 只 是 用 来 调用 print_r() 函数 。 必 须 为 该 函数 提供 变量 名 。 例 如 ， 变 量 名 为 
$_SERVER， 这 是 PHP 中 一 个 特殊 的 变量 。$_SERVER 变 量 储 存 关 于 服务 器 的 所 有 数据 : 名 称 、 操 
作 系 统 、 当 前 用 户 名 、Web 服 务 器 应 用 程序 (Apache、Abyss、IIS 等 ) 的 信息 ， 以 及 其 他 信息 。 
它 也 提供 所 执行 的 PHP 脚 本 的 信息 : 脚本 名 称 、 在 服务 器 上 保存 的 位 置 等 。 

请 注意 ， 必 须 严格 按照 变量 全 大 写 的 样式 正确 输入 $_SERVER。 








(4) 完成 HIML 页 面 ; 
</pre> 

</body> 

</html> 


(5) 将 文件 保存 为 predefined.php 并 上 传 至 服务 器 (或 者 保存 在 本 机 的 目录 下 )， 并 且 在 
Web 浏 览 器 上 测试 (参见 图 2-1)。 再 次 强调 ， 请 记 住 必须 通过 URL 来 运行 所 有 的 PHP 脚 本 。 

















Array 
( 
[HTTP_ HOST] => phpvqs4:8888 
[HTTP USER AGENT] => Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10.6; 
[HTTP ACCEPT] => text/html,application/xhtml+xml,application/xml;q=0 
[HTTP ACCEPT LANGUAGE] => en-us,en;q=0.5 
[HTTP ACCEPT ENCODING] => gzip,deflate 
[BHTTP_ ACCEPT CHARSET] => IS0-8859-1,utf-8;q=0.7,*;q=0.7 
[HTTP KEEP ALIVE] => 115 
[HTTP CONNECTION] => keep-alive 
[PATH] => /usr/bin:/bin:/usr/sbin:/sbin 
[SERVER_SIGNATURE] => 
[SERVER SOFTWARE] => Apache 
[SERVER NAME] => phpvqs4 
[SERVER ADDR] => 127.0.0.1 
[SERVER PORT] => 8888 
[REMOTE ADDR] => 127.0.0.1 
[DOCUMENT ROOT] => /Users/larryullman/Sites/phpvqs4 
[SERVER ADMIN] => youeexample .com 
[SCRIPT FILENAME] => /Users/larryullman/Sites/phpvqs4/predefined.php 
[REMOTE PORT] => 51149 
[GATEWAY INTERFACE] => CGI/1.1 
[SERVER PROTOCOL] => HTTP/1.1 
[REQUEST METHOD] => GET 
[QUERY STRING] => 
[REQUEST URI] => /predefined.php 
[SCRIPT NAME] => /predefined.php 
[PHP_ SELF] => /predefined.php 
[REQUEST TIME] => 1289831848 
[argv] => Array 
( 
) 


[argc] => 0 

















图 2-1 该 脚本 打印 了 $_SERVER 变 量 的 值 ， 它 是 有 关 服 务 器 和 PHP 脚 本 的 值 的 主 列表 


(6) 如 果 有 可 能 , 将 文件 保存 在 另外 一 台 可 运行 PHP 的 计算 机 或 者 服务 器 上 , 然后 在 Web 浏 览 
器 上 重新 运行 脚本 (参见 图 2-2)。 
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Array 
( 
[HTTP HOST] => larryullman.com 
[HTTP_ USER AGENT] => Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10 6 5; en-us 
[HTTP ACCEPT] => application/xm]l,application/xhtml+xm]l,text/html;q=0.9,text/ 
[HTTP_ ACCEPT LANGUAGE] => en-us 
[HTTP_ ACCEPT ENCODING] => gzip, deflate 
[HTTP CONNECTION] => keep-alive 
[PATH] => /sbin:/usr/sbin:/bin:/usr/bin:/usr/X11R6/bin 
[SERVER_SIGNATURE] => 
Apache/2.0.63 (CentOs) Server at larryullman.com Port 80 





[SERVER SOFTWARE] => Apache/2.0.63 (CentOs) 
[SERVER NAME] => larryullman.com 

[SERVER ADDR] => 207.58.187.78 

[SERVER PORT] => 80 

[REMOTE ADDR] => 71.58.97.51 


[DOCUMENT ROOT] => Wp I A "8 -httpdocs 
[SERVER ADMIN] => Larry@DMCInsights.com 
[SCRIPT FILENAME] => 7 TAGAG SE a -httpdocs/predefined.php 


[REMOTE PORT] => 44766 
[GATEWAY_INTERFACE] => CGI/1.1 
[SERVER PROTOCOL] => HTTP/1.1 
[REQUEST METHOD] => GET 
[QUERY STRING] => 
[REQUEST URI] => /predefined.php 
[SCRIPT NAME] => /predefined.php 
[PHP_ SELF] => /predefined.php 
[REQUEST_ TIME] => 1289832083 

) 


2-2 不 同 的 服务 器 在 运行 predefined.php 时 ， 页 面 将 显示 不 同 的 结果 (请 同 图 2-1 进 行 对 比 ) 














内 














ww 提示 

口 像 前 面 示例 中 这 样 打印 出 变量 的 值 是 进行 调试 的 主要 手段 ， 这 是 因为 通常 问题 出 在 变量 
中 没有 保存 想 要 的 值 。 脚 本 有 时 不 会 按 预 期 工作 ， 因 为 某 些 变量 的 值 可 能 和 你 想 的 不 一 
样 ， 因 此 确认 变量 的 实际 值 非常 重要 。 

口 如 果 不 使 用 <pre></pre> HTML 标签， 结果 就 会 如 图 2-3 中 显示 的 那样 非常 凌乱 。 




















Array ( [HTTP_HOST] => phpvqs4:8888 [HTTP_USER_AGENT] => Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10.6; en-US; rv:1.9.2.12) Gecko 
;gq=0.8 [HTTP_ACCEPT_LANGUAGE] => en-us.en;q=0.5 [HTTP_ACCEPT_ENCODING] => gzip,deflate [HTTP_ACCEPT_CHARSET] =>1 

[HTTP_CACHE_CONTROL] => max-age=0 [PATH] => /usr/bin:/bin:/usr/sbin:/sbin [SERVER_SIGNATURE] => [SERVER_SOFTWARE] => Ap 
[REMOTE_ADDR] => 127.0.0.1 [DOCUMENT_ROOT] => /Users/larryullman/Sites/phpvqs4 [SERVER_ADMIN] => you@example.com [SCRIP 
[GATEWAY _INTERFACE] => CGI/1.1 [SERVER_PROTOCOL] => HITP/1.1 [REQUEST_METHOD] => GET [QUERY_STRING] =>[REQUE 
[REQUEST_TIME] => 1289832176 [argv] => Array ( ) [argc] => 0) 











图 2-3 HTML 预 格式 化 标签 能 够 避免 在 使 用 print_r() 时 出 现 这 样 令 人 难以 看 清 的 页 面 
(请 同 图 2-1 和 图 2-2 进 行 对 比 ) 





2.2 变量 语 法 


现在 已 经 初步 了 解 了 变量 ， 接 下 来 将 进一步 探讨 有 关 变 量 的 内 容 。 在 先前 的 示例 中 使 用 了 
PHP 中 $_SERVER 这 个 预定 义 变量 。 在 掌握 了 正确 的 语法 后 ， 也 可 以 自己 创建 变量 。 创 建 正确 的 
变量 名 ， 必 须 遵循 以 下 规则 
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口 所 有 的 变量 名 必须 以 美元 符号 ($) 开头 ; 
口 在 美元 符号 后 的 第 一 个 字符 必须 为 字母 (A ~Z, a~z) 或 者 下 划 线 (_), 不 能 使 用 数字 ，; 
口 变量 名 剩 下 的 部 分 可 以 包含 任何 数量 的 字母 、 数 字 和 下 划 线 组 合 ; 
口 变量 名 中 不 能 出 现 空格 (通常 使 用 下 划 线 进行 文字 分 隔 ); 
口 变量 名 必须 唯一 ; 
口 变量 名 是 区 分 大 小 写 的 ! 这 意味 着 $variable 和 $Variable 是 截然 不 同 的 变量 ， 因 此 将 
两 个 变量 进行 如 此 相似 的 命名 是 非常 不 明智 的 。 
最 后 也 是 最 重要 的 一 点 : PHP 中 的 变量 名 是 区 分 大 小 写 的 。 使 用 错误 的 大 小 写字 母 是 导致 bug 
发 生 的 常见 原因 。 例 如 ， 如 果 在 前 面 脚本 中 使 用 $_servez 或 $S_Sserver， 你 可 能 会 收 到 错误 信息 
或 什么 也 看 不 到 (参见 图 2-4)。 


















































Notice: Undefined variable: server in /Users/larryullman/Sites/phpvqs4/predefined.php on line 13 














WN 








2-4 ”错误 地 拼写 变量 名 (包括 大 小 写 错 误 )， 可 能 出 现 不 需要 的 或 无 法 预计 的 结果 


本 书 提供 以 下 建议 来 帮助 读者 尽 可 能 减少 bug 发 生 的 可 能 性 。 

口 变量 名 全 部 使 用 小 写字 母 。 

口 让 变量 名 具有 描述 性 (例如 ，$first_name 就 比 $fn 好 得 多 )。 

口 使 用 注释 来 说 明 变量 的 用 途 (参看 脚本 2-2) ， 虽 然 看 上 去 这 似乎 有 些 多 余 。 
口 最 重要 的 ， 保 持 一 致 的 命名 模式 。 

表 2-1 列 出 了 一 些 有 效 的 变量 

表 2-2 列 出 了 一 些 无 效 的 变量 名 及 其 违反 的 规则 。 














表 2-1 PHP 中 有 效 的 变量 表 2-2 ”PHP 中 无 效 的 变量 名 
变 量 名 变 量 名 原 
$sfirst_name sfirst name 包含 空格 
$person $first.name 包含 英文 句点 
$addressl first_name 没有 以 $ 开 头 
$_SERVER $1laddress $ 后 的 第 一 个 字符 不 能 为 数字 














脚本 2-2 这 段 代 码 显示 了 如 何 使 用 注释 来 说 明 变量 的 用 途 ， 通 常 详 细 地 进行 注释 总 是 比 不 用 注 
释 要 好 得 多 


1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Variables and Comments</title> 

7 </head> 

8 <body> 

9 


<?php // 脚本 2-2 
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Syear = 2011; // 当前 年 。 
$june_avg = 88; // 6 月 份 的 平均 气温 。 
Spage_title = 'Weather Reports'; // 页 面 的 标题 。 





Dy 
</body> 
</html> 
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Vv 提示 

口 与 其 他 语言 不 同 ，PHP 通 常 不 需要 在 使 用 变量 前 声明 或 者 初始 化 变量 。 换 句 话 说， 可 以 在 
定义 变量 类 型 之 前 先 引 用 。 尽 管 如 此 ， 最 好 还 是 不 要 这 样 做 。 本 书 所 编写 的 脚本 将 尽力 
做 到 先 定义 变量 后 再 使 用 。 

口 变量 命名 的 惯例 主要 有 两 种 ， 可 以 根据 所 使 用 的 不 同 摘 述 方式 来 选择 。 它 们 是 所 谓 的 驼 
峰 (camel-hump) 方式 〈 使 用 大 写字 母 对 单词 进行 分 隔 ， 例 如 SFizstName) 和 下 划 线 方 
式 (例如 $first_name)。 本 书 将 在 示例 中 使 用 后 一 种 方式 。 











2.3 ”变量 类 型 


本 书 将 介绍 3 种 变量 类 型 . 数值 型 、 字 符 串 型 和 数组 型 。 这 里 只 作 简 要 的 介绍 , 后 面 几 章 (第 
4 章 、 第 5 章 和 第 7 章 ) 将 对 详细 介绍 它们 。 第 4 种 变量 类 型 一 一 对 象 ， 将 在 附录 B 中 介绍 ， 而 不 包 
含 在 本 书 正 文中 , 因为 它 是 一 个 特殊 的 主题 , 对 于 初学 者 来 说 难度 太 大 。 事实 上 , 在 我 编著 的 PPP 
5 Advanced: Visual QuickPro Guide (Peachpit 2007 年 出 版 ) 中 ， 使 用 了 超过 150 页 的 篇 幅 来 讲述 这 
个 主题 ， 却 仅仅 只 涉及 了 这 个 主题 的 基础 部 分 。 


2.3.1 数值 


从 技术 上 讲 ， 在 PHP 中 数值 类 型 分 为 两 种 : 整 型 和 浮 点 型 (也 被 称 为 双 精 度 浮 点 型 )。 因 为 
PHP 对 于 变量 的 处 理 方式 相对 随意 ， 所 以 不 会 阻止 对 两 种 类 型 的 数值 进行 同样 的 处 理 。 同 样 ， 为 
了 表述 清晰 本 书 将 对 两 种 类 型 的 区 别 作 简明 介绍 。 

数值 的 第 一 种 类 型 一 一 整 型 ， 同 整数 一 样 。 它 们 可 以 是 正 数 或 者 负数 ， 但 是 不 能 包含 有 分 数 
或 者 小 数 部 分 。 带 有 小 数 点 的 数值 〈 就 像 1.0) 是 浮 点 型 。 必 须 使 用 浮 点 型 的 数值 来 表示 分 数 ， 
由 于 在 PHP 中 表示 分 数 的 唯一 方式 就 是 将 之 转换 为 等 价 的 小 数 ， 因此 1 二 用 1.25 表 示 。 表 2-3 列 举 


了 一 些 有 效 数 值 的 示例 以 及 它们 对 应 的 类 型 ， 表 2-4 列 举 了 无 效 的 数值 及 其 违反 的 规则 。 
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表 2-3 ”PHP 中 的 有 效 数值 表 2-4 PHP 中 的 无 效 数值 

数 值 类 型 数 值 原 因 

1 整 型 1/3 包含 斜 线 

1.0 浮 点 型 1996a 包含 字母 

1972 整 型 08.02.06 包含 多 个 小 数 点 
19.72 浮 点 型 

-1 整 型 

-1.0 浮 点 型 

Vv 提示 


口 下 节 就 可 以 了 解 到 ， 可 以 将 无 效 的 数值 用 引号 引用 从 而 转换 为 有 效 的 字符 串 。 
2.3.2 ”字符 日 

一 个 字符 串 是 一 对 单 引 号 (' ) 或 者 双 引 号 (") 引用 的 任意 数量 的 字符 。 字 符 串 能 够 包含 字 
母 、 数 字 、 符 号 和 空格 的 任何 组 合 ， 它 还 能 够 包含 变量 。 

下 面 是 有 效 字 符 串 的 示例 ; 


"Hello, world!" 
"Hello, S$first name!" 








i 
'Hello, world! How are you today?' 
"08.02.06" 


"1996" 





最 后 一 个 示例 是 一 个 空 字符 囊 : 一 个 没有 包含 任何 字符 的 字符 串 。 

简 言 之 ,创建 字符 串 ， 就 是 用 引号 引用 一 些 字符 。 但 在 使 用 时 ， 你 可 能 会 遇 到 一 些 问题 ， 例 
如 : 

"I said, "How are you?"" 

这 个 示例 有 些 不 好 理解 , 本 书 在 第 1 章 中 介绍 打印 HTML 代 码 时 提 到 过 这 样 的 问题 。 当 PHP 
遇 到 第 2 个 引号 时 ， 它 认为 这 是 字符 串 的 结束 标志 ， 接 下 来 的 文本 (How 及 其 后 的 字符 ) 将 会 
引发 错误 。 如 前 所 述 ， 可 以 在 字符 串 里 使 用 引号 ,但 是 需要 在 其 之 前 放置 反 斜 线 (\) 将 其 转 
义 : 

"I said, \"How are you?\"" 

这 样 就 可 以 告诉 PHP 这 两 个 引号 是 字符 串 的 值 ， 而 不 是 字符 串 开 启 和 关闭 的 指示 符 。 

可 以 使 用 不 同 的 引号 类 型 来 解决 这 个 问题 : 


'I said, "How are You?1" 
"I said, 'How are you?'" 
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v 提 示 

吕 请 注意 '1996" 这 个 示例 ， 将 整 型 字符 放置 在 引号 中 就 转换 成 为 字符 串 型 。 从 本 质 上 说 ， 
字符 串 中 包含 的 字符 1996 同 数值 1996 (未 被 引用 ) 相等 。 这 是 一 个 难以 看 出 的 区 别 ， 而 

且 在 代码 中 不 会 引发 问题 。 这 是 因为 可 以 使 用 字符 串 1996 进 行 数学 运算 ， 就 如 同 使 用 数 2 
值 1996 一 样 。 

口 在 第 1 意 中 ， 本 书 介绍 了 如 何 使 用 被 双 引号 引用 的 \n 字 符 创建 一 个 新 行 。 转 义 引 号 将 打印 
引号 ， 转 义 n 将 打印 新 行 ， 转 义 r 可 以 创建 一 个 辕 车 符 ， 转 义 t 将 向 HTML 源 代码 中 插入 一 
个 制 表 符 。 

D 理解 字符 串 、 变 量 和 单 双 引号 对 于 使 用 PHP 编 程 来 说 非常 重要 。 出 于 这 个 原因 ,本 书 将 在 
2.5 节 专门 阐述 这 个 主题 。 
































2.3.3 ”数组 


本 书 将 在 第 7 章 对 数组 进行 详细 的 讨论 ， 这 里 只 作 简 明 的 介绍 。 字 符 串 和 数值 类 型 (都 被 称 
之 为 标量 ) 只 包含 一 个 值 ， 数 组 可 以 被 赋予 多 个 值 ， 可 以 认为 数组 是 值 的 列表 。 换 名 话说 ， 可 以 
在 一 个 数组 中 放置 多 个 字符 串 和 /或 数值 。 

数组 使 用 键 来 创建 和 检索 它们 保存 的 值 ， 其 结果 构成 为 一 系列 键 - 值 对 ， 看 上 去 像 是 一 个 两 
列 的 数据 表 。 有 趣 的 是 ， 在 PHP 中 的 数组 结构 非常 灵活 ， 它 的 键 和 值 都 可 以 使 用 数值 或 者 字符 串 
类 型 。 数 组 甚至 不 必 在 此 方面 保持 一 致 性 〈 在 第 7 章 接触 到 具体 示例 时 ， 你 就 会 明白 这 里 所 说 的 
意思 )。 

PHP 有 两 种 数组 类 型 ,区别 在 于 键 的 格式 。 如 果 一 个 数组 使 用 数值 作为 键 (参见 表 2-5) 那么 
它 就 是 索引 数组 ， 如 果 它 使 用 字符 串 作为 键 (参见 表 2-6) ， 那 么 就 是 关联 数组 。 在 任何 一 种 情况 
下 ， 数 组 中 的 值 都 可 以 是 任何 变量 类 型 (字符 串 、 数 值 或 者 其 他 类 型 )。 





















































Vv 提示 

口 数组 的 键 又 被 叫做 索引 ， 这 两 个 术语 意思 相同 。 

口 一 个 数组 可 以 并 且 经 常 包含 其 他 的 数组 ， 这 种 数组 叫做 多 维 数组 。 
口 PHP 中 所 谓 的 关联 数组 ， 在 Per 和 Ruby 中 叫做 数列。 














表 2-5 索引 数值 表 2-6 ”关联 数值 
键 值 键 值 
0 Don VT Vermont 
1 Betty NH New Hampshire 
和 2 Roger IA Iowa 
3 Jane PA Pennsylvania 
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2.4 为 变量 赋值 


在 为 变量 赋值 时 ， 可 以 忽略 变量 的 类 型 ， 使 用 等 号 〈=) 给 变量 赋值 。 因 此 ， 等 号 被 称 作为 
赋值 运算 符 ， 并 且 变 量 在 赋值 运算 符 左边 ， 而 被 赋 的 值 在 右边 。 例 如 : 








snumber = 1; 
$sfloating number = 1.2; 
sstring = "Hello, world!"; 


每 一 行 都 是 一 个 完整 的 赋值 语句 ( 即 一 个 可 执行 的 动作 )， 每 一 句 结尾 都 要 有 一 个 分 号 。 
打印 变量 值 ， 可 以 使 用 print 国 数 : 


print Snumber 
print $string; 


如 果 希 望 将 上 下 文中 的 变量 值 打 印 出 来 ,可 以 将 变量 名 放置 在 被 打印 的 字符 串 中 ,只 要 使 用 
双 引 号 将 它们 正确 引用 即 可 (参见 图 2-5): 


print "Number is Snurmber'" ; 
print "String is $string"; 








Number is 1 
string is Hello, world! 














图 2-5 ”打印 两 个 变量 值 的 结果 


print 的 这 种 用 法 只 适用 于 标量 ( 单 值 ) 变量 类 型 ， 即 数值 和 字符 串 型 变量 ， 而 数组 和 对 象 
是 比较 复杂 的 变量 类 型 ， 不 能 仅 使 用 print 函 数 (参见 图 2-6): 


print "_SERVER is $_SERVER"; 





























_SERVER is Array 














图 2-6 ”用 print 语 句 数 组 类 的 复杂 变量 类 型 ， 不 能 得 到 你 想 要 的 结果 


你 可 以 看 到 , print_r() 可 以 处 理 这 些 非 标 量 类 型 。 在 本 章 的 后 面 ,你 还 会 学 到 其 他 处 理 方 
法 。 无论 是 处 理 标 量变 量 还 是 非 标 量变 量 , 在 脚本 出 现 问 题 时 ,打印 出 它们 的 值 总 是 非常 好 的 调 
试 技术 。 

由 于 变量 类 型 并 不 是 锁定 的 (PHP 被 称 为 弱 类 型 语言 )， 因 此 它们 可 以 在 运行 的 时 候 被 修改 : 


$svariable 
$svariable 


如 果 现 在 打印 $variable 的 值 ， 结 果 将 是 Greetings。 下 面 的 脚本 更 好 地 诠释 了 为 变量 赋 
值 并 访问 这 些 值 的 方法 。 





























二 
"Greetings"; 





为 变量 赋值 并 进行 访问 














(1) 在 文本 编辑 器 或 IDE 中 创建 新 的 PHP 脚 本 ， 命 名 为 variables .php (参看 脚本 2-3)。 





24 0U000050 33 





脚本 2-3 ”这 个 脚本 定义 了 一 些 基 本 的 变量 ， 并 且 将 它们 的 值 打印 出 来 


3 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

见 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

三 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 
5 

6 

7 

8 











<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Variables</title> 


</head> 
<body> 
9 <?php // 脚本 2-3 - variables.php 
10 
11 // 地 址 : 
12 $street = "100 Main Street"; 
13 S$city = "State College"; 
14 S$state = "PA"; 
15 S$zip = 16801; 
16 


17 // 打印 地 址 : 
18 Print "<p>The address is:<br/>$street <br/>S$Scity $state S$zip</p>"; 


19 
20 ?> 
21 </body> 


22 </html> 
(2) 创建 初始 的 HTML 标 签 。 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Variables</title> 

</head> 

<body> 

(3) 开始 编写 PHP 代 码 : 

<?php // 脚本 2-3 - variables.php 


(4) 定义 一 些 数值 和 字符 串 变 量 : 


sstreet = "100 Main Street"; 
Scity = "State College"; 
$sstate = "PA"; 

$szip = 16801; 


这 些 代码 创建 了 一 些 不 同 的 字符 串 和 数值 类 型 的 变量 。 字符 串 型 使 用 引号 来 进行 定义 , 并且 
个 变量 名 都 遵循 了 正确 的 命名 规则 。 

说 记 ， 每 个 语句 都 要 使 用 分 号 作为 结尾 ， 变 量 名 是 区 分 大 小 写 的 。 

(5) 打印 变量 ; 

print "<p>The address is:<br/>$street <br/>s$city $state $zip</p>"; 


这 里 使 用 一 个 print 语 句 访问 所 有 的 变量 。 完 整 的 需 打印 的 字符 串 (由 文本 、HTML 标 签 和 
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变量 旨 明成 ) 被 双 引 号 所 引用 。 HTML 代 码 中 的 <br/> 标 签 使 文本 在 浏览 器 中 以 多 行 的 形式 显 
注意 ， 在 换行 标签 中 额外 的 空白 和 和 斜 线 是 为 了 符合 XHTML 的 要 求 ) 。 
a : 


> 
</body> 
</htmil> 


示 (请 


07) 将 文件 保存 为 variables .php， 并 上 传 到 服务 器 上 (或 者 保存 在 本 机 合适 的 目录 下 )， 
然后 在 Web 浏 览 器 上 进行 测试 (参见 图 2-7)。 








AMO Variables 





The address is: 
100 Main Street 
State College PA 16801 


























图 2-7 ”图 为 一 些 被 赋值 的 变量 ， 在 打印 时 带 有 上 下 文 





w 提示 


口 如 果 在 运行 此 脚本 时 看 到 语法 解析 错误 (参见 图 2-8), 很 可 能 是 因为 遗漏 了 分 号 , 或 者 是 
有 未 配对 的 引号 。 





























Parse error: syntax error, unexpected T_VARIABLE in 
/Users/larryullman/Sites/phpvqs4/variables.php on line 13 











图 2-8 ”如 你 所 见 ， 语 法 解析 错误 是 PHP 中 最 常见 的 错误 类 型 ， 它 们 通常 由 缺少 分 号 和 

未 配对 的 引号 或 圆 括 号 引起 

口 如 果 一 个 变量 的 值 没 有 被 打印 出 来 ,或 者 看 到 Undefined variable 错 误 提 示 (参见 图 2-9)， 

很 可 能 是 因为 两 次 拼写 变量 名 不 一 致 所 造成 的 拼写 错误 。 

口 如 果 看 到 空白 页 ， 很 可 能 是 出 了 某 些 错误 ， 同 时 PHP 中 display_errors 的 设置 被 设 为 
off。 参 看 第 3 章 中 的 相关 内 容 。 


AMA Variables 






































Notice: Undefined variable: Street in 
/Users/larryullman/Sites/phpvqs4/variables.php on line 18 


The address is: 








State College PA 16801 




















图 2-9 ”Undefined variable 错 误 意味 着 使 用 了 一 个 没有 值 的 变量 (变量 未 被 定义 ) ， 
这 可 能 由 拼写 错误 和 大 小 写 不 一 致 引起 


2.5 理解 引号 


既然 已 经 对 变量 的 基础 知识 以 及 如 何 创 建 变量 有 所 了 解 ， 本 书 将 开始 阐述 引号 的 重要 概念 。 
如 大 多 数 编程 语言 一 样 ，PHP 允 许 使 用 单 引 号 (') 和 双 引 号 〈")， 但 是 它们 将 导致 截然 不 同 的 
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结果 。 理 解 这 一 点 是 非常 重要 的 ， 因 此 在 接 0 中 将 演示 使 用 这 两 种 引号 的 区 别 。 
a 使 用 单 引号 引用 的 内 容 将 照 字 面 意思 进行 处 理 ， 而 被 双 引 号 引用 的 内 容 需 要 
进行 推断 。 也 就 是 说 ， 双 引号 引用 的 变 ee 正如 在 脚本 2-3 中 看 到 的 那样 。 


但 单 引号 引用 的 变量 名 不 会 被 赤 代 。 2 


这 条 规则 在 PHP 里 用 到 引号 的 情况 中 普遍 适用 ， 包 括 创建 字符 串 变量 和 使 用 print 函 数 。 下 
面 是 解释 这 条 规则 的 最 佳 示 例 。 


信使 用 引号 


(1) 在 文本 编辑 器 或 者 IDE 中 创建 一 段 新 的 PHP 脚 本 ， 命 名 为 auotes .php (参看 脚本 2-4) 。 
脚本 2-4 这 段 脚本 清楚 地 诠释 了 使 用 不 同类 型 的 引号 将 导致 的 不 同 结果 


























<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

之 "http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

总 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 

7 

8 

9 








<title>Quotes</title> 
</head> 
<body> 
<?php // 脚本 2-4 - quotes.php 


11 // 这 里 既 可 使 用 单 引 号 ， 又 可 使 用 双 引 号 : 
12 S$first name = 'Larry'; 
13 $last name = "Ullman"; 


15 // 这 里 单 引号 和 双 引 号 的 作用 不 同 : 
16 S$namel = !$Eirst_name $last name'; 
17 S$name2 = "$first name $last name"; 





19 // 这 里 单 引 号 和 双 引 号 的 作用 不 同 : 

20 Print "<hl>Double Quotes</hl><p>namel is $namel <br/> 
21 name2 is $name2</p>"; 

22 

23 print '<hl>Single Quotes</hl><p>namel is $namel <br/> 
24 name2 is $name2</p>'; 

25 

.6 7 之 

27 </body> 

28 </html> 


(2) 创建 初始 HTML 标 签 : 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Quotes</title> 

</head> 
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<body> 

(3) 开始 编写 PHP 代 码 : 

<?php // 脚本 2-4 - quotes.php 
(4) 创建 两 个 字符 串 变量 : 


$sfirst name = 'Larry'; 
$last name = "Ullman"; 


为 这 两 种 变量 使 用 单 引 号 或 者 双 引 号 没有 任何 区 别 , 因为 对 于 字符 串 都 会 取 其 字面 值 。 但 是 ， 








如 果 在 这 里 使 用 自己 的 名 字 (完全 可 以 ) 并 且 你 的 名 字 里 包含 一 个 撤 号 ,那么 要 么 需要 使 用 双 引 
号 引用 ,要么 将 撒 号 转 义 并 用 单 引号 引用 : 


$last_name 
$last_name 


(5) 用 Sfirst-name 和 $last-name 变 量 创建 Snamel 和 和 $name2 变 量 : 


"O'Toole"; 
'O\'Toole'; 


snamel 
$name2 


在 这 里 ， 使 用 不 同 的 引号 类 型 就 会 导致 不 同 的 结果 。$namel 变 量 会 按 字面 处 理 ， 结 果 为 


'$Sfirst name $last_ name'; 
"Sfirst name $last name"; 






































$first_name $last_name, 因为 这 里 没有 进行 推断 。 相反 , Sname2 的 结果 将 是 Larry Ullman,， 


也 就 是 预想 中 的 结果 。 


值 ， 


在 print 语 句 中 的 HTML 代 码 让 它们 在 浏览 器 中 更 易于 阅 。 区 99 si 
读 , 并 且 每 个 语句 都 分 两 行 打印 变量 的 值 , 这 让 人 更 容易 接受 。 

(7) 完成 PHP 代 码 段 和 HTML 页 面 ; Double Quotes 

?> namel is Sfirst_name Slast_name 

</body> name2 is Larry Ullman 


在 本 机 合适 的 目录 下 )， 然 后 在 Web 浏 览 器 上 进行 测试 (参见 图 




















(6) 使 用 两 种 引号 类 型 打印 变量 : 


print "<hl>Double Quotes</hl> <p>namel is S$namel <br/> 
name2 is $name2</p>"; 
print '<hl>Single Quotes</hl> <p>namel is S$namel <br/> 
name2 is $name2</p>'; 


不 同类 型 的 引号 再 次 产生 了 不 同 的 结果 。 第 一 个 print 语 句 打 印 出 Sname1 和 Sname2 变 量 的 
而 第 二 个 Print 语句 则 直接 打印 出 Sname1 和 Sname2。 











</html> 


(8) 将 文件 保存 为 quotes .php 并 上 传 至 服务 器 (或 者 保存 ”| Single Quotes 








namel is Snamel 












































2-10 ) 。 name2 is Sname2 
V 提示 图 2-10 不 同 的 引号 ( 单 引 号 和 
器 吕 成 妈 [: 米 或 二 双 引 号 ) 指明 需要 打印 
口 如 果 对 于 两 种 引号 的 区 别 仍然 感到 迷惑 ， 坚持 使 用 双 引 的 是 变量 的 名 称 还 是 变 


号 将 是 个 保险 的 做 法 。 量 的 值 
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口 可 以 认为 使 用 单 引 号 更 为 可 取 , 因为 PHP 不 需要 通过 搜索 字符 串 来 查找 变量 。 这 个 规则 更 
多 的 只 是 一 种 技巧 一 一 对 性 能 影响 非常 小 ， 可 以 忽略 不 计 。 

D 创建 新 行 (\n)、 回 车 (\r) 和 制 表 符 (\t) 的 快捷 方式 都 必须 在 双 引 号 内 使 用 以 达到 预 
期 的 效果 。 
D 请 记 住 ， 并 不 是 总 需要 使 用 引号 的 。 当 所 赋 的 值 为 数值 类 型 ， 或 者 只 需要 打印 变量 时 ， 
可 以 省 略 它们 : 

Snum = 2; 

Drint. Sum> 
































2.6 回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
comy/forumy/ 。 


2.6.1 回顾 


口 $_SERVER 是 什么 类 型 的 变量 ? 

口 变量 名 是 以 什么 打头 的 ? 

口 变量 名 的 第 一 个 字符 可 以 使 用 哪些 字符 ? 在 第 一 个 字符 后 面 可 以 使 用 哪些 字符 ? 
口 变量 名 区 分 大 小 写 吗 ? 

口 什么 样 的 变量 是 0 0 ? 举 出 一 些 不 同 的 标量 变量 类 型 。 举 出 一 个 非 标量 变量 类 型 。 
口 赋值 操作 符 是 什么 ? 

口 检查 变量 时 ， 本 章 介 绍 的 调试 技术 是 什么 ? 

口 双 引 号 和 单 引 号 在 使 用 上 有 什么 不 同 之 处 ? 








2.6.2 ”实践 

口 创建 另 一 个 PHP 脚 本 ， 定 义 一 些 变量 并 打印 它们 的 值 。 尽 量 使 用 不 同 标量 类 型 的 变量 。 

D 创建 一 个 PHP 脚 本 ， 在 HTML 代码 中 打印 一 些 变量 的 值 。 为 了 加 大 难度 ， 可 以 使 用 PHP 和 
变量 创建 链接 标签 或 图 片 标签 。 











HTML 表 单 和 PHP 








本 章 内 容 

口 创建 简单 的 表单 

口 选择 表单 的 提交 方法 
口 使 用 PHP 接 收 表单 数据 
口 显示 错误 

口 错误 报告 

口 向 页 面 手 动 发 送 数据 
口 回顾 和 实践 


第 2 音 简 要 地 介绍 了 变量 。 读 者 会 经 常 创建 自己 的 变量 ， 也 会 经 常 使 用 同 HTML 表 单 关 联 的 
变量 。 表 单 是 当今 Web 网 站 的 基本 单位 ， 能 够 实现 注册 和 登录 系统 、 搜 索 功 能 、 在 线 购物 等 功能 
和 特性 。 其 至 在 最 基本 的 网 站 里 都 能 找到 HTML 表 单 的 逻辑 。 同 时 ， 利 用 PHP 就 能 非常 简单 地 歼 
得 和 处 理由 HTML 表 单 生成 的 数据 。 

出 于 以 上 方面 的 考虑 ， 本 章 将 介绍 创建 HTML 表 单 以 及 如 何 利用 PHP 访 问 表 单 中 数据 的 基础 
知识 。 同时, 本 章 还 会 介绍 实际 PHP 编 程 中 几 个 关键 性 的 概念 , 其 中 包括 如 何 处 理 脚 本 中 的 错误 。 


3 1 创 建 简 单 的 表 单 Please complete this form to submit your feedback: 
Name: [Mr 网] 
先 创建 一 个 反馈 页 面 以 获取 用 户 的 称呼 、 名 称 、 电 | vow、 


子 邮件 地 址 、 反 馈 和 备注 (参见 图 3-1)， 这 个 页 面 将 用 
在 本 章 的 HTML 表 单 示例 中 。 这 里 需要 针对 上 面 的 要 求 
创建 所 需 的 字段 。 这 些 用 来 生成 表单 的 代码 都 包含 在 开 
局 和 关 闭 的 标签 中 Comments: 


Send My Feedback 

















Response: This is... |© excellent O okay O boring 














<form> 
form elements 
</form> 


form 标 签 指明 代码 中 一 个 表单 的 开头 和 结尾 。 每 个 表单 中 的 元 素 都 必须 处 于 这 对 标记 中 。 
开启 form 标 记 还 包含 一 个 action 属 性 ， 它 用 来 指明 将 表单 数据 提交 至 哪个 页 面 。 这 是 在 创建 表 








图 3-1 在 本 章 示例 中 使 用 的 HTML 表 单 
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单 时 需要 着 重 考 虑 的 问题 。 在 本 书 中 ，action 属 性 通常 将 指向 PHP 脚 本 : 


则 导 








<form action="somepage.php"> 


在 创建 下 一 个 表单 之 前 ， 请 重新 简短 回顾 一 下 XHTML。 如 第 1 章 所 述 ，XHTML 的 一 些 规 
改 它 与 HTML 有 截然 不 同 的 语法 。 对 初学 者 来 说 ,代码 需要 全 部 使 用 小 写字 母 , 并 且 每 个 











标签 属性 都 必须 加 引号 引用 。 而 且 ,每 个 标签 都 必须 关闭 ,那些 没有 关闭 形式 的 标签 (如 input) 
必须 要 在 末尾 处 添加 空格 和 和 斜 线 。 因 此 ， 在 HIML 中 可 以 这 样 书写 





<INPUT TYPE=TEXT NAME=address SIZE=40> 


但 是 XHTML 中 需要 这 样 书写 


的 ， 





<input type= "text" name="address" size="40" /> 

本 书 希望 这 个 简短 的 解释 能 帮助 读者 消除 在 下 面 这 上 段 代码 中 由 XHTML5| 起 的 疑惑 。 
HIML 和 XHTML 中 的 每 个 表单 元 素 的 名 字 必 须 唯 一 。HIML 和 XHTML 的 命名 规则 是 一 致 
只 能 使 用 字母 、 数 字 和 下 划 线 (_)。 总 而 言 之 ， 名 字 应 当 符 合 逻辑 且 描 述 性 强 。 


之 创建 基本 的 HTML 表 单 








(GD 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 文档 ， 命 名 为 feedback.html (参看 脚本 3-1): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Feedback Form</title> 

</head> 

<body> 

<!-- 脚本 3-1 - feedback.html --> 

<div><p>Please complete this form to submit your feedback:</p> 





脚本 3-1 这 个 HIML 页 面 中 有 一 个 带 有 几 个 不 同类 型 输入 方式 的 表单 








工 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

用 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Feedback Form</title> 

7 </head> 

8 <body> 

9 <!-- 脚本 3-1 - feedback.html --> 





10 <div><p>Please complete this form to submit your feedback:</p> 


12 <xform action="handle form.php"> 


13 

14 <p>Name: <select name="title"> 

15 <option value="Mr.">Mr.</option> 
16 <option value="Mrs.">Mrs.</option> 


17 <option value="Ms.">Ms.</option> 
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18 </select> <input type="text" name="name" size="20" /></p> 

19 

20 <p>Email Address: <input type="text" name="email" size="20" /></p> 
21 

22 <p>Response: This is... 

23 <input type="radio" name="response" value="excellent" /> excellent 
24 <input type="radio" name="response" value="okay" /> okay 

25 <input type="radio" name="response" value="boring" /> boring</p> 
26 

ye <p>Comments: <textarea name="comments" rows="3" cols="30"></textarea></p> 
28 

29 <input type="submit" name="submit" value="Send My Feedback" /> 

30 

31 </form> 

32~ ,</div> 

33 </body> 


34 </html> 

(2) 添加 表单 的 开启 标签 : 

<form action="handle_ form.php"> 

这 个 表单 标签 指明 将 把 表单 提交 至 handle_form.php 页 面 ， 它 同 这 个 HIML 页面 在 相同 的 
目录 下 。 如 果 和 希望 更 加 明晰 ， 可 以 在 PHP 脚 本 中 使 用 完整 的 URL 路 径 〈( 例 如， 
http:/www.example.com/handle_form.php ) 。 

(3) 为 填写 人 名 添加 一 个 选择 菜单 以 及 一 个 文本 输入 框 : 


<p>Name: <select name="title"> 
<option value="Mr.">Mr.</option> 
<option value="Mrs.">Mrs.</option> 
<option value="Ms.">Ms.</option> 
</select> <input type="text" 
name="name" size="20" /></p> 


这 个 用 以 输入 人 名 的 部 分 将 由 两 个 元 素 组 成 (参见 图 3-1) 。 第 一 个 元 素 是 一 个 选项 为 常用 称 
谓 的 下 拉 菜 单 ， 选 项 为 : ML 、Mrs. 和 Ms。 每 个 列举 在 select 标 签 中 的 选项 都 可 供用 户 进行 选择 
(参见 图 3-2)。 第 二 个 元 素 是 一 个 基本 文本 框 ， 用 来 填写 完整 名 称 。 











Name: 





v 






Mr. eT 





~ Mrs. 
Email + 











图 3-2 ”该 选择 元 素 创建 了 一 个 带 有 选项 的 下 拉 菜 单 














(4) 添加 一 个 文本 框 用 来 填写 用 户 的 电子 邮件 地 址 : 


<p>Email Address: <input type="text" name="email" size="20" /></p> 


(5) 为 反馈 添加 单 选 按钮 


<p>Response: This is... 
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<input type="radio" name="response" value="excellent" /> excellent 

<input type="radio" name="response" value="okay" /> okay 

<input type="radio" name="response" value="boring" /> boring</p> 

这 段 HTML 代 码 创 建 了 3 个 单 选 按钮 (可 以 点 击 的 圆圈 ， 参 见 图 3-1)。 由 于 它们 的 name 值 相 
同 ， 因 此 一 次 只 能 选择 3 个 中 的 1 个 。 在 XHTML 规则 中 ， 除 了 值 以 外 代码 都 用 小 写字 母 形式 ， 并 
且 需 要 在 每 个 input 标 签 的 末尾 处 使 用 一 个 额外 的 空格 和 和 斜 线 以 关闭 该 标签 。 

(6) 添加 一 个 textarea 用 来 记录 备注 : 

<p>Comments: <textarea name="comments" rows="3" cols="30"></textarea></p> 

textarea 能 够 比 文 本 框 提 供 更 多 的 空间 以 供用 户 填写 备注 等 文字 。 但 是 文本 框 可 以 对 用 户 
能 够 输入 的 信息 数量 进行 限制 ， 这 是 textarea 所 不 能 做 到 的 (确切 地 说 是 在 没有 使 用 JavaScript 
的 情况 下 )。 当 创建 表单 时 ， 需 要 依照 从 用 户 获 取信 息 的 不 同方 式 选 择 相 应 的 input 控 制 类 型 。 

请 注意 ，textarea 没 有 关闭 标签 。 

0) 添加 提交 按钮 

<input type="submit" name="submit" value="Send My Feedback" /> 

提交 元 素 的 value 属 性 将 显示 在 Web 浏 览 器 中 相应 的 按钮 上 (参见 图 3-1)。 也 可 以 使 用 例如 
Gol! 或 Enter 的 字样 。 

(8) 关闭 表单 : 

</form> 

(9) 完成 页 面 : 


</div> 
</body> 
</html> 


(10) 将 页 面 保存 为 feedback.html 并 在 浏览 器 上 查看 。 

由 于 这 是 个 HTML 页 面 而 不 是 PHP 脚 本 , 因此 可 以 在 自己 的 计算 机 上 用 Web 浏 览 器 直接 查看 。 

Vv 提示 

口 请 注意 ， 在 这 里 使 用 HTML 扩 展 名 ( .html) 是 因为 它 是 一 个 标准 的 HTML 页 面 (而 不 是 
PHP 页 面 )。 也 可 以 使 用 .php 扩 展 名 ， 这 不 会 造成 任何 问题 ,即便 是 页 面 里 并 不 包含 任何 
PHP 代 码 。( 请 记 住 ，PHP 标 签 外 面 的 代码 一 一 <?php 和 ?> 一 一 都 将 被 视 作 HTML。) 

口 确认 action 属 性 正确 地 指向 在 服务 器 上 实际 存在 的 文件 ， 否 则 表单 将 不 能 被 正常 提交 。 

在 这 种 情况 下 , 需要 指明 应 该 把 该 表单 提交 至 handle_form.php, 它 同 feedback .html 
页 位 于 相同 的 目录 下 。 

D 在 这 个 示例 中 ， 一 个 HTML 表 单 被 手动 编码 的 HTML 创建 

页 面 应 用 程序 中 创建 表单 (例如 ，Adobe Dreamweaver) 。 

口 在 即将 到 来 的 HTML 5 中 ， 增 添 了 一 些 非常 受 欢迎 的 新 表单 元 素 ， 如 Email、url 和 


number, 


















































如 果 愿 意 的 话 ， 还 可 以 在 Web 
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3.2 ”选择 表单 的 method 


经 验 丰富 的 HTML 开 发 人 员 将 会 注意 到 ， 这 个 反馈 表单 缺少 一 件 东西 : 初始 表单 标记 中 没有 
method 属 性 。 这 个 属性 是 用 来 告诉 服务 器 如 何 将 数据 从 表单 传送 至 处 理 脚 本 的 。 
methodq 有 两 种 方法 可 供 选 择 : GET 和 POST。 许 多 HTML 程 序 员 并 不 完全 清楚 它们 之 间 的 区 
别 ， 以 及 何 时 该 用 哪个 。GET 和 POST 之 间 的 区 别 在 于 将 信息 从 表单 向 处 理 信息 的 脚本 传递 的 方 
式 。GET 将 所 有 的 信息 聚集 起 来 并 作为 URL 的 一 部 分 进行 传递 ， 而 POST 并 不 让 用 户 看 到 传递 的 
信息 。 例 如 ， 当 提交 表单 时 ， 如 果 使 用 GET 方 式 ，URL 看 上 去 就 可 能 像 这 样 : 
http:/www.example.com/page.php? some_var=some_value&age=20&... 
而 使 用 POST 方 式 时 ， 最 终 用 户 将 只 能 看 到 : 
http:/www.example.com/page.php 
当 在 两 种 方式 中 进行 选择 时 ， 请 考虑 下 面 的 4 个 因素 : 
口 使 用 GET 方 式 时 ， 传 送 的 信息 量 有 限 ， 
口 GET 方 式 公开 地 将 输入 的 信息 向 处 理 脚 本 传送 (这 意味 着 ， 诸 如 密码 这 样 的 信息 都 将 会 
被 任何 注视 Web 浏 览 器 的 人 看 到 ， 这 将 引发 严重 的 安全 问题 ) ， 
口 使 用 GET 方 式 的 表单 所 创建 的 页 面 能 够 添加 为 书签 ， 但 是 POST 方式 却 不 可 以 ; 
口 如 果 用 户 试图 重 载 通过 POST 访问 的 页 面 ， 他 就 会 收 到 提示 信息 (参见 图 3-3)， 但 重 载 通 
过 GET 访 问 的 页 面 就 不 会 收 到 提示 信息 。 


















































Confirm 
To display this page, Firefox must send information that 
will repeat any action (Such as a search or order 


confirmation) that was performed earlier. 




















图 3-3 如 果 用 户 刷新 PHP 脚 本 ， 且 该 脚本 的 数据 是 通过 POST 方式 发 送 的 ， 
他 就 会 收 到 系统 的 确认 信息 (具体 的 信息 取决 于 使 用 的 浏览 器 ) 

一 般 来 说 ， 从 服务 器 上 请 求 信息 时 ,使 用 GET 请 求 。 几 乎 所 有 用 于 搜索 的 页 面 都 会 使 用 GET 
( 当 你 使 用 搜索 引擎 时 ,看 一 下 它 的 URL), 就 像 那 些 分 页 显示 结果 的 页 面 一 样 ( 比 如 分 类 浏览 产 
品 )。 相 反 ，POST 一 般 用 于 触发 基于 服务 器 的 行为 ， 比 如 提交 一 张 联系 表 (发 送 Email) 或 提交 
某 个 博客 的 评论 (评论 添加 到 数据 库 和 该 博客 的 页 面 上 )。 

尽管 GET 方 法 也 有 一 些 有 用 的 技术 (参看 3.6 市 ), 但 是 本 书 将 专门 使 用 POST 方 法 来 处 理 表 单 。 

这 回 脚本 添加 方法 

(GD 在 文本 编辑 器 或 者 IDE 中 打开 feedback.ntml 文 件 (参看 脚本 3-1)。 

(2) 在 初始 表单 标签 中 添加 methodq= "post" (参看 脚本 3-2 中 第 12 行 )。 


脚本 3-2 添加 值 为 POST 的 method 属 性 以 完成 表单 


J <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
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为 


步 








总 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Feedback Form</title> 

7 </head> 

8 <body> 

9 <!-- 脚本 3-2 - feedback.html --> 

10 <div><p>Please complete this form to submit your feedback:</p> 

站 证 

12 <xform action="handle form.php" method="post"> 

3 

4 <p>Name: <select name="title"> 

5 <option value="Mr.">Mr.</option> 

16 <option value="Mrs.">Mrs.</option> 

7 <option value="Ms.">Ms.</option> 

8 </select> <input type="text" name="name" size="20" /></p> 

9 
20 <p>Email Address: <input type="text" name="email" size="20" /></p> 
21 
22 <p>Response: This is... 
2 六 <input type="radio" name="response" value="excellent" /> excellent 
24 <input type="radio" name="response" value="okay" /> okay 
25 <input type="radio" name="response" value="boring" /> boring</p> 
26 
2.7 <p>Comments: <textarea name="comments" rows="3" cols="30"></textarea></p> 
28 
29 <input type="submit" name="submit" value="Send My Feedback" /> 

30 

SL /DOT 

32 </div> 

33 </body> 


34 </html> 
表单 的 method 属 性 告诉 浏览 器 如 何 将 表单 数据 发 送 到 接收 脚本 。 因 为 在 提交 表单 时 包含 很 
多 数据 (包括 备注 )， 并且 因为 用 户 并 不 会 将 结果 页 面 添 加 为 书签 ， 所 以 在 这 里 使 用 POST 方 法 更 


合理 。 








(3) 保存 脚本 ， 并 在 Web 浏 览 器 上 重新 加 载 。 
应 该 养 成 在 修改 了 脚本 之 后 ， 立 即 在 Web 浏 览 器 中 重新 加 载 页 面 的 习惯 。 重 新 加 载 页 面 这 一 
常常 会 被 忽略 ， 导 致 你 对 更 改 后 的 页 面 没有 任何 变化 而 迷惑 不 解 。 


(4) 查看 页 


Vv 提示 





面 的 源 代码 以 确定 所 有 的 元 素 都 出 现在 页 面 上 并 且 具 有 正确 的 属性 (参见 图 3-4)。 








D 在 讨论 方法 的 时 候 ，GET 和 POST 都 使 用 了 大 写字 母 以 引起 注意 。 但 是 ， 在 脚本 中 的 表单 
内 ， 要 使 用 POST 以 保持 同 XHTML 的 一 致 性 。 不 要 为 这 种 不 一 致 感到 担心 (如果 发 现存 
在 这 样 的 情况 ) ，methoa 属 性 并 不 区 分 大 小 写 。 
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EN 









2/ /WwW W3.O0rg/TR/xhtmili/DTD/xhtmll~tr nal.dtd”"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 


<head> 
<meta http-equiv="Content-Type”" content="text/html; charset=utf-8"/> 
<title>Feedback Form</title> 
</head> 
<body> 
<!1--~ Script 3.2 -~ feedback.html --> 
<div><p>Please complete this form to submit your feedback:</p> 
<form action="handle form.php" method="post"> 
<p>Name: <select name= "title"> 
<option value="Mr.">Mr.</option> 
<option value="Mrs.">Mrs.</option> 
<option value="Ms.">Ms.</option> 
</select> <input type="text" name="name" size="20" /></p> 
<p>Email Address: <input type="text" name= "email” size="20" /></p> 
<p>Response: This is... 
<input type="radio" name="response”" value="excellent" /> excellent 
<input type="radio" name="response" value="okay" /> okay 
<input type="radio" name="response" value="boring" /> boring</p> 
<p>Comments: <textarea name="comments" rows="3" cols="30"></textarea></p> 
<input type="submit" name="submit" value="Send My Feedback" /> 
</form> 
</div> 
</body> 
</html> 


图 3-4 ”在 表单 中 , 很 多 重要 的 信息 (如 action 和 methoqa 的 值 或 者 元 素 的 名 称 ) 都 只 能 
在 HTML 源 代码 中 看 到 


3.3 使 用 PHP 接收 表单 数据 


前 两 布 我 们 创建 了 一 个 基本 的 HTML 表 单 ， 现 在 需要 编写 PHP 脚 本 ， 它 将 接收 和 处 理 表 单数 
据 。 在 这 个 示例 中 ，PHP 脚 本 将 重复 用 户 在 表单 中 输入 的 内 容 。 第 4 章 将 介绍 如 何 获 得 这 些 信息 
并 将 它们 保存 到 数据 库 中 ， 如 何 通过 电子 邮件 发 送 这 些 信息 以 及 如 何 将 信息 写 入 文件 ， 等 等 。 

访问 被 提交 的 表单 数据 ， 需要 查阅 第 2 章 中 关于 预定 义 变量 的 内 容 ， 在 那里 已 经 介绍 了 预定 
义 变量 $_SERVER。 这 个 PHP 脚 本 用 来 处 理 表 单数 据 而 引用 的 特定 变量 可 能 是 $_GET 或 $_POST 中 
的 一 个 。 如 果 HTML 表 单 使 用 GET 方 法 , 那么 将 能 够 通过 $_GET 找 到 提交 的 表单 数据 。 如 果 HTML 
表单 使 用 POST 方法 ， 那 么 被 提交 的 表单 数据 将 存储 在 $_PosT 中 。 

$_GET 和 $_POST 除 了 是 预定 义 变量 (也 就 是 不 需要 创建 ) 外 ， 还 是 数组 和 一 个 特殊 的 变量 
类 型 。 这 就 意味 着 每 个 变量 都 可 能 包含 大 量 的 值 。 不 能 这 样 使 用 数组 (参见 图 2-6): 

print $_POST; // 无 法 工作 ! 

相反 ,为 了 访问 一 个 特殊 的 值 ， 必 须 引 用 数组 的 索引 或 者 键 。 在 第 7 章 将 对 此 进行 详细 介 
但 是 这 个 前 提 真 的 非常 简单 。 从 一 个 name 属 性 的 值 为 something 的 表单 元 素 开 始 : 


<input type= "text" name="something" /> 


现在 ,假设 表单 使 用 POST 方 法 ， 在 表单 元 素 中 输入 的 值 在 $_POST['something'] 中 将 是 
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有 效 的 : 


print $_POST['something']; 


遗憾 的 是 , 这 里 存在 一 个 小 问题 : 当 在 双 引 号 中 使 用 单 引 号 引用 键 时 ， 将 引发 解析 错误 ( 参 





图 3-5 ) 。 


print "Thanks for saying:S$_POST['something']" 


Parse error: syntax error, unexpected Se 
T_ENCAPSED_AND_WHITESPACE., expecting T_STRING or 

T_VARIABLE orT_ NUM_STRING in /Users/larryullman/Sites 
/phpvqs4/handle_form.php on line 19 


图 3-5 ”这 个 难看 的 解析 错误 是 由 试图 在 双 引 号 中 使 用 $_POSTI' something']3 引 | 起 的 
有 几 种 方法 可 以 避免 这 样 的 问题 。 在 本 章 中 ,将 选用 语法 最 为 简单 的 一 种 方法 : 只 是 先 把 这 





























个 特殊 的 $_PoST 元 素 赋值 给 另 一 个 变量 


$something = $_POST['something']; 
print "Thanks for saying: $something"; 


在 新 的 PHP 脚 本 中 执行 这 个 信息 前 最 后 还 有 两 点 需要 注意 的 是 。 首先 , $_POST 是 区 分 大 小 


写 的 : 它 必 须 严格 地 像 你 所 看 到 的 这 样 输入 : 一 个 美元 符号 、 一 个 下 划 线 ， 然 后 是 所 有 大 写 的 


字母 。 其 次 ， 在 先前 的 示例 中 ，$_POST 的 索引 








必须 严格 地 同 提供 给 表单 元 素 的 name 值 匹配 。 


(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 文档 ， 命 名 为 handle_form.php (参看 脚本 3-3): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Your Feedback</title> 

</head> 

<body> 











脚本 3-3 ”此 脚本 显示 通过 引用 关联 的 $_POST 变 量 向 脚本 提交 的 表单 数据 


. <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

久 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 

5 








<head> 
<meta http-equiv="Content-Type" 
content="text/html; charset=utf-8"/> 

6 <title>Your Feedback</title> 

7 </head> 

8 <body> 

9 <?php // 脚本 3-3 handle_form.php 

1 

1 


Fa 


// 该 页 面 从 feedback.html 接 收 数据 。 
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// 将 会 接收 : 标题 、 姓 名 、Email、 反 映 、 评 论 并 用 $_POST 提 交 。 


// 为 新 的 变量 赋值 : 

$title = $ POST['title']; 
Sname = $_ POST['name']; 
$response = $ POST['response']; 

$comments = $_POST['comments']; 

// 打印 接收 到 的 数据 : 

print "<p>Thank you, $title $name, for your comments.</p> 
<p>You stated that you found this example to be 'S$response' and 
added: <br/>$comments</p>"; 


?> 
</body> 
</html> 


(C) 添加 PHP 开 启 标 签 和 一 些 注释 : 


从 学 
A 
// 


向 


php // 脚本 3-3 handle_form.php 
该 页 面 从 feedqback .htm1 接 收 数据 。 
将 会 接收 : 标题 、 姓 名 、Email、 反 映 、 评 论 并 用 $_POST 提 交 。 


脚本 添加 注释 是 为 了 让 它 的 执行 目的 更 加 清晰 。 在 feedqpback .html 页 面 中 指明 了 数据 被 











发 送 的 目的 地 (通过 action 属 性 ), 同时 注释 从 相反 的 方向 进行 益 述 (脚本 从 哪里 接收 这 些 数据 )。 
(3) 用 接收 到 的 数据 为 新 的 变量 赋值 : 


St 
Sn 
$r 
SC 


当 














+tLTe, SS SG :POSTE" titLe",s 

ame = $_POST['name']; 

esponse = $_POST['response']; 
omments = $_POST['comments']; 


表单 使 用 POST 方法 时 ， 提 交 的 数据 能 够 在 $_PosT 数 组 中 找到 。 可 使 用 语法 





$_POST['name_attribute_value'] 访 问 这 个 独立 的 值 。 这 些 操 作 将 不 会 受到 表单 元 素 类 型 


(文本 
为 


匡 、 下 拉 框 、 复 选 框 等 ) 的 影响 。 
了 能 够 更 加 容易 地 在 第 (4) 步 print 语 句 中 使 用 这 些 值 ， 在 这 里 每 个 变量 都 将 被 赋予 新 的 











值 。 本 书 没 有 对 $_POST['email'] 或 $_POST['submit'] 做 任何 处 理 ， 如 果 需 要 的 话 ， 你 可 以 
加 上 一 些 处 理 。 
(4) 将 用 户 信息 打印 出 来 : 


pr 
<p 


int "<p>Thank you, $title $name, for your comments.</p> 
>You stated that you found this example to be 'Sresponse' and added: 


<br/>$scomments</p>"; 
这 个 print 语 名 在 上 下 文中 使 用 了 4 个 变量 ， 用 来 向 用 户 显示 脚本 接收 了 什么 数据 。 
(5) 关闭 PHP 代 码 片 段 并 且 完 成 HTML 页 面 : 


全 
志 办 
</ 


body> 
html> 


(6) 将 脚本 保存 为 handle_form.php。 
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注意 文件 名 必须 与 action 属 性 的 值 完全 相同 。 

(7) 将 脚本 上 传 至 服务 器 〈 如 果 在 自己 的 计算 机 上 安装 了 PHP， 也 可 以 保存 在 本 机 适当 的 目 
录 下 ) ， 确 保 保 存在 同 feedback .html 相 同 的 目录 下 。 

(8) 通过 URL (http://something) 加 载 feedback .html。 

必须 通过 URL 来 加 载 HTML 表 单 ， 这 样 当 向 PHP 脚 本 提交 表单 时 ，PHP 脚 本 也 能 够 通过 URL 
运行 。 广 意 : PHPD DDDODDDURLDDODO 

忘记 通过 URL 加 载 表 单 是 初学 者 常常 会 犯 的 错误 。 

(9) 填写 表单 (参见 图 3-6)， 完 成 后 提交 (参见 图 3-7)。 






































Please complete this form to submit your feedback: 


Name: [ Mr. | Larry Ullman 


Email Address: larry@example.com 





Response: This is. @ excellent © okay © boring 





No problems so far! 


Comments: 


Send My Feedback 


图 3-6 ”用 户 在 HTML 表 单 中 输入 的 所 有 内 容 都 应 该 通过 handle_form.php 脚 本 
显示 在 Web 浏 览 器 中 (参见 图 3-7) 









































Thank you, Mr. Larry Ullman, for your comments. 


You stated that you found this example to be 'excellent and added: 
No problems so far! 




















图 3-7 ”这 是 在 第 1 章 中 讨论 过 的 print 语 句 的 男 一 种 应 用 ， 它 创建 了 第 一 个 动态 
生成 的 网 页 








如 果 看 到 空白 页 ， 阅 读 3.4 往 ， 了 解 如 何 显 示 可 能 发 生 的 错误 。 
如 果 看 到 一 个 错误 通知 〈 参 见 图 3-8) 或 者 看 到 一 个 变量 在 打印 输出 的 时 候 没 有 值 ， 很 有 可 
能 是 因为 表单 元 素 的 name 值 或 者 $_POST 数 组 的 索引 拼写 错误 (或 者 没有 将 表单 填写 完整 )。 



































Notice: Undefined index: Name in /Users/larryullman/Sites 
/phpvqs4/handle_form.php on line 16 





Thank you, Mr. , for your comments. 


图 3-8 ”如 果 以 某 种 形式 使 用 了 并 不 存在 的 变量 ， 那 么 就 会 出 现 如 图 所 示 的 通知 。 
在 本 例 中 ， 发 生 错 误 是 因为 将 $_POST['name'] 写 成 $_POST['Name'] 了 


























Vv 提示 
口 如 果 希 望 向 PHP 脚 本 传递 预 设 值 , 可 以 在 HTML 表单 中 使 用 文本 输入 框 的 隐藏 类 型 。 例 如 ， 
在 表单 标签 中 插入 代码 行 : 
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<input type="hidden" name="form page" 
value="feedback.html" /> 


将 创建 一 个 值 为 feedback.html 的 变量 ， 它 在 处 理 脚 本 的 时 候 将 调用 $_POSTI['this_ 
page']。 
口 请 注意 ， 单 选 钮 和 选择 菜单 变量 的 值 是 基于 被 选择 项 的 value 属 性 (例如 ， 单 选 钮 的 
excellent 值 )， 这 对 于 复 选 框 也 适用 。 对 于 文本 框 ， 变 量 的 值 就 是 用 户 输入 的 内 容 。 
口 如 果 handle_form.php 脚 本 在 提交 的 字符 串 中 显示 出 多 余 的 斜 线 , 请 查看 “Magic Quote” 
框 注 中 提供 的 解释 和 解决 方案 。 
无 论 表 单 使 用 什么 nethod， 你 都 可 以 在 预定 义 的 $_REQUEST 变 量 中 访问 表单 数据 。 然 而 ， 
$_GET 和 $_POST 更 明确 ， 所 以 更 可 取 。 



































Magic Quote 


DPHPUUUUUUUUDMagic QuoteUUUUUUUUUUUUUOO0O0O0000 
DUOUOUUOOOUOOO0O0O0ODO00D0 MagicQuoteuUUOOOUUOUO0O0O0O0O0000 
OOOOOOODOOOOOOODOOO0O0OD Edlike more informationl DUDUUUNdIike more 
information[| 

D00000000000000000000000000000000000UUU00 
DUDUPHPUDUD MagicQuoteuUOUOUODUOPHPUUOODOOOOOOOUOUOOOOOOOOO0O000 
stripslashes DD DOD0UUUUUUUUUUU handle form.php OUUUUUUUUO 


Scomments = stripslashes($_ POST ['comments']); 


DUOUD 


$comments = $_POST['comments']; 


Og 
D000000000000000000U000UUU Magic Quotel 


3.4 显示 错误 


在 调试 PHP 脚 本 时 出 现 的 首要 问题 是 ， 在 这 里 不 一 定 能 看 到 发 生 的 错误 。 在 Web 服 务 器 
上 安装 PHP 之 后 ， 它 将 在 默认 的 安全 配置 下 运行 ， 处 理 数 据 的 方式 、 表 现 的 性 能 等 都 将 在 这 
个 环境 中 进行 。 默 认 设置 中 的 一 项 是 不 显示 发 生 的 任何 错误 。 换 句 话 说 ，display_errors 
设置 处 于 关闭 状态 (参见 图 3-9)。 在 这 种 情况 下 ， 如 果 脚 本 发 生 错 误 ， 看 到 的 将 是 一 个 空白 
页 (这 是 全 新 安装 的 PHP 的 标准 设置 ， 大 多 数 托管 的 公司 都 会 把 Gisplay_errors 设 置 为 开 
启 状态 )。 

不 在 活动 的 网 站 里 显示 错误 的 原因 是 为 了 规避 一 些 安 全 风险 。 简 单 地 说 ，PHP 的 错误 显示 通 
常会 给 公众 提供 过 于 详尽 的 信息 。 但 是 , 作为 开发 人 员 来 说 , 必须 能 够 查看 这 些 错误 以 修正 它们 。 
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display_errors Of Off 
display startup orors lm on 
doc root om nov 
docrer ent oe nov 
docref root eue love 
enabie dl ma 
error_append_string no value 
error log /Applications/MAMP/logs 
php_error.log Iphp_error.log 
enor propond string oe lovee 
error_reporting 32767 32767 














图 3-9 ”运行 phpinfo() 脚 本 (如 脚本 1-2)， 查 看 在 服务 器 中 display_errors 的 设置 状态 


为 了 能 够 显示 PHP 错 误 ， 可 以 进行 以 下 操作 : 
口 开启 display_errors 设 置 (参看 附录 A 以 获取 更 多 信息 ); 
口 为 某 些 单独 的 脚本 开启 aisplay_errors 设 置 。 

当 开 发 一 个 网 站 时 ,到 目前 为 止 第 一 个 选项 是 首选 。 但 是 ,这 个 选项 仅仅 是 针对 拥有 服务 器 
管理 员 权 限 的 用 户 。 任 何人 都 可 以 通过 在 脚本 中 包含 以 下 代码 行 来 选用 第 二 个 选项 : 

ini_set ('display_errors', 1); 

ini_set () 函数 允许 脚本 临时 覆盖 PHP 配 置 文件 中 的 设置 。 在 这 个 示例 中 ， 将 开启 
display_errors 设 置 ， 在 这 里 用 数字 1 代表 开启 状态 。 

尽管 任何 人 都 可 以 使 用 第 二 种 方式 , 但 是 其 不 足 之 处 在 于 : 如 果 脚 本 发 生 某 些 特定 类 型 的 错 
误 (将 在 第 4 章 中 进行 讨论 )， 将 不 会 被 执行 。 因此， 这 行 代码 也 就 不 会 被 执行 ,并且 这 种 特定 错 
误 ， 以 及 任何 阻止 脚本 运行 的 因素 都 将 导致 出 现 一 个 空白 页 面 。 














瀛 显示 错误 的 步骤 








(1) 在 文本 编辑 器 或 IDE 中 打开 handle_form.php。 
@2) 在 PHP 代 码 的 第 一 行 ， 输 入 以 下 代码 (参看 脚本 3-4): 


ini_set ('display_errors', 1); 


脚本 3-4 ”向 PHP 脚 本 添加 的 这 个 代码 段 将 开启 Gisplay_errors 设 置 ， 它 将 使 发 生 的 所 有 错误 
都 被 显示 出 来 








<?php // 脚本 3-4 - handle_form.php #2 


1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://ww.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Your Feedback</title> 

7 </head> 

8 <body> 

9 

1 
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11 ini set ('display errors', 1); 
// 让 我 从 错误 中 学 到 知识 ! 

12 

13 // 该 页 面 从 feedback.html 接 收 数据 。 

14 // 将 会 接收 : 标题 、 姓 名 、Email、 反 映 、 评 论 并 用 $_POST 提 交 。 

15 

16 // 为 新 的 变量 赋值 : 

17 Stitle = $_POST['title']; 

18 S$name = $_POST['name']; 

19 Sresponse = $_POST['response'l]; 

20 S$comments = $_POST['comments']; 

21 

22 // 打印 接收 到 的 数据 : 

23 print "<p>Thank you，Stitle Sname， for 
your comments.</p> 

24 <p>You stated that you found this example to be 'Sresponse' and added: 
<br/>$comments</p>"; 

2:5 

26… 了 六 

27 </body> 

28 </html> 


该 行 代码 将 告诉 PHP 希 望 看 到 发 生 的 任何 错误 。 需 要 在 PHP 代 码 段 首先 对 它 进 和 
剩 下 的 PHP 代 码 都 遵从 这 个 新 的 设 定 。 
(3) 将 文件 保存 为 handle_form.php。 


(4) 上 传 文件 至 Web 服 务 器 ， 并 在 Web 浏 览 器 上 进行 测试 (参见 图 3-10 和 图 3-11 





Please complete this form to submit your feedback: 





Name: [Mr A Larry UlIman 





Email Address: 


调用， 以 便 


)。 
































Response: This is... OQ excellent O okay O boring 
Notice: Undefined index: response in /Users/larryullman/Sites 
/phpvqs4/handle_form.php on line 19 
Comments: Thank you, Mr. Larry Ullman., for your comments. 
(Send My Feedback ) You stated that you found this example to be " and added: 
图 3-10 再 次 尝试 打开 这 个 表单 图 3-11 现在 发 生 的 任何 错误 都 被 显示 出 来 。 生 成 的 





如 果 结 果 页 中 没有 任何 错误 ， 脚 本 将 像 先前 那样 








通知 表明 引用 的 表单 元 素 没有 值 


运行 。 如 果 之 前 运行 表单 时 看 到 了 空白 页 ， 








现在 将 可 能 看 到 图 3-11 中 显示 的 消息 。 如 果 看 到 这 样 的 错误 ， 很 可 能 是 因为 表单 元 素 的 名 称 拼 写 
错误 、$_PoST 数 组 的 索引 拼写 错误 ， 或 者 是 没有 完整 地 填写 表单 。 
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Vv 提示 

口 确保 在 进行 任何 困难 的 脚本 调试 时 ，display_errors 都 处 于 开启 状态 。 如 果 在 计算 机 
上 安装 了 PHP, 本 书 强烈 建议 在 学 习 时 开启 其 中 的 display_errors 设 置 (请 再 次 参考 附 
录 A)。 

口 如 果 运 行 PHP 脚 本 时 看 到 了 空白 页 ， 也 需要 检查 HTML 源 代码 以 查找 错误 或 其 他 问题 。 

D ini_set () 国 数 只 能 用 来 更 改革 些 特定 的 设置 。 参 看 PHP 手 册 以 了 解 更 多 细节 。 

口 请 记 住 , aisplay_errors 只 控制 是 否 要 向 Web 浏 览 器 发 送 错误 消息 。 它 并 不 产生 错误 也 
不 会 阻止 错误 的 发 生 。 

如 果 在 name 表 单元 素 后 面 漏 掉 等 号 ， 也 会 出 现 问题 : 


<input name"something" /> 

















3.5 ”错误 报告 


需要 了 解 与 display_errors 有 关 的 另 一 个 PHP 配 置 问题 是 错误 报告 。 在 PHP 中 有 11 种 不 同 
的 错误 ， 而 在 版 本 6 中 ， 另 外 还 有 4 种 用 户 定义 的 类 型 (本 书 将 不 做 介绍 )。 表 3-1 中 列举 了 最 重要 
的 4 种 常见 错误 类 型 ， 并 对 它们 进行 了 描述 和 举例 。 
表 3-1 PHP 错 误 类 型 















































类 型 描 述 示 例 

通知 非 致命 性 错误 ， 是 或 者 不 是 表明 有 问题 存在 引用 一 个 没有 值 的 变量 

警告 非 致命 性 错误 ， 通 常 表明 有 问题 存在 国 数 误 用 

办 析 错 误 由 语法 错误 导致 的 致命 性 错误 缺少 分 号 ， 或 者 引号 、 圆 括号 和 人 花 括号 
省 误 一 般 性 的 致命 错误 内 存 分 配 问 题 











有 两 种 方法 设置 PHP 报 告 哪些 错误 。 第 一 种 方法 是 在 PHP 的 配置 文件 中 调整 error_ 
reporting() 级 别 (参见 附录 A)。 如 果 你 使 用 的 是 自己 的 PHP 服 务 器 ,在 开发 脚本 时 可 能 需要 
调整 服务 器 的 全 局 设置 。 

第 二 种 方法 是 在 脚本 中 使 用 error_reporting() 函数 , 该 函数 带 有 一 个 数字 或 者 一 些 吕 





DD (未 使 用 引号 引用 的 字符 串 ， 具 有 预先 指定 的 意义 ) 来 调整 级 别 。 表 3-2 列 出 了 比较 重要 的 一 
些 常量 。 
表 3-2 错误 报告 常量 
名 称 名 称 
E_NOTICE E_ERROR 
E_WARNING E_ALL 
E_PARSE E_STRICT 


使 用 这 些 信息 ， 可 以 向 脚本 添加 下 面 的 代码 : 


error_reporting (0); 
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error_reporting (E_ALL); 
error_reporting (E ALL & ~E NOTICE); 


第 一 行 代码 指明 不 需要 报告 任何 错误 。 第 二 行 代 码 要 求 报告 所 有 的 错误 。 最 后 一 个 示例 说 明 
需要 看 到 除 notice 之 外 的 所 有 错误 消息 。 请 牢记 ， 调 整 这 些 设 置 不 会 阻止 或 者 产生 错误 ， 它 只 是 
影响 是 否 报 告 错误 。 

通常 使 用 最 高 级 的 错误 报告 对 于 开发 和 测试 PHP 脚 本 来 说 是 最 好 的 选择 。 可 以 声明 希望 看 到 
所 有 的 错误 以 及 strict 错 误 报告 : 

error_reporting (E_ALL | FE_STRICT); 

E_ALL 设 置 并 不 包含 E_STRIcCT， 这 就 是 为 什么 这 行 代 码 表明 所 有 的 错误 信息 将 被 显示 ， 或 
者 〈 竖 线 ， 称 为 管道 符 ) strict 错 误 被 显示 的 原因 。 后 面 的 设置 对 错误 提供 了 更 进一步 的 报告 ， 但 
是 也 可 能 会 成 为 在 PHP 未 来 版 本 中 引发 notice 提 示 问 题 的 原因 。 让 我 们 在 handle_form.php 页 面 
中 应 用 这 个 设置 。 


忆 调整 错误 报告 









































(1) 在 文本 编辑 器 或 IDE 中 打开 handle_form.php (参看 脚本 3-4)。 
(2) 在 ini_set () 的 下 1 行 , 添加 下 面 的 代码 (参看 脚本 3-5): 


error_reporting (E_ALL | E_STRICT); 














脚本 3-5 ”调整 脚本 的 错误 报告 级 别 可 以 对 次 在 的 和 已 经 存在 的 问题 提供 或 多 或 少 的 反馈 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 
5 <meta http-equiv="Content-Type" 
content="text/html; charset=utf-8"/> 
6 <title>Your Feedback</title> 
7 </head> 
8 <body> 
9 <?php // 脚本 3-5- handle_form.php #3 


11 ini set ('display_ errors', 1); 
// 让 我 从 错误 中 学 到 知识 ! 

12 error reporting (E ALL | E STRICT); 
// 显示 所 有 可 能 出 现 的 问题 ! 


14 // 该 页 面 从 feedback.html 接 收 数据 。 
15 // 将 会 接收 : 标题 、 姓 名 、Email、 反 映 、 评 论 并 用 $_POST 提 交 。 


17 // 为 新 的 变量 赋值 。 

18 Stitle = $_POST['title']; 

19 S$name = $_POST['name']; 

20 Sresponse = $_POST['response'l]; 
21 S$comments = $_POST['comments']; 
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23 // 打印 接收 到 的 数据 : 

24 print "<p>Thank you, $title $name, for your comments.</p> 

25 <p>You stated that you found this example to be 'Sresponse' and added: 
<br/>$scomments</p>"; 


26 
7 最 
28 </body> 


29 </html> 


(3) 将 文件 保存 为 handle_form.php。 
(4) 在 启用 了 PHP 的 服务 器 上 将 文件 保存 至 适当 的 目录 , 并 通过 提交 表单 在 Web 浏 览 器 上 进行 
测试 (参见 图 3-12 和 图 3-13)。 











Please complete this form to submit your feedback: 


Name: [ Ms， 并 Blankenship 


Email Address: ida@example.edu 








Response: This is... O excellent O okay @ boring 





Enough already! 


Comments: 








(Send My Feedback ) 











图 3-12 再 次 测试 表单 





Thank you, Ms. Blankenship, for your comments. 


You stated that you found this example to be boring and added: 
Enough already! 


图 3-13 ”试验 的 结果 (如果 填写 完整 ， 并 且 没 有 任何 编程 错误 ) 
此 时 此 刻 ， 如 果 表 单 填写 完整 ， 并 且 $_PosT 的 索引 同 表 单元 素 的 name 值 精确 匹配 ， 将 看 不 


到 任何 错误 发 生 (正如 图 中 显示 的 那样 )。 如 果 有 问题 存在 ， 包 含 任何 潜在 的 问题 (由 于 有 
E_STRICT)， 它 们 都 应 该 显示 出 来 并 产生 错误 报告 。 





















































Vv 提示 

口 PHP 手 册 列 举 了 所 有 的 错误 报告 级 别 ， 但 是 在 这 里 列举 出 来 的 都 是 其 中 最 重要 的 错误 
类 型 。 

口 本 书 中 的 代码 都 是 使 用 最 高 级 别 的 错误 报告 (E_ALL |E_sTRICT) 测试 的 。 


3.6 向 页 面 手动 发 送 数据 


本 章 的 最 后 一 个 示例 和 其 他 的 主题 比 起 来 有 些 跑题 ,但 就 如 何 使 用 PHP 处 理 表单 数据 提供 了 
额外 的 补充 。 正 如 在 3.2 节 中 讨论 的 那样 ， 如 果 表 单 使 用 GET 方 法 ，URL 将 会 像 这 样 : 
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http:/www.example.com/page.php? some_var=some_value&age=20&... 

向 接收 页 面 (在 这 里 是 page .php) 发 送 了 一 系列 诸如 name=value 的 信息 对 ， 它 们 中 的 每 
一 个 都 被 与 号 (&) 分 隔 开 。 整 个 序列 由 一 个 问号 开始 〈 紧 随 在 正在 处 理 的 脚本 名 称 之 后 ) 。 

为 了 用 这 种 方式 访问 传递 给 页 面 的 值 ， 可 以 使 用 $_GET 变 量 。 正 如 使 用 $_PosT 那 样 ， 引 用 
这 个 特定 的 名 称 作 为 $_GET 的 索引 。 在 这 个 示例 中 ，page.php 接 收 了 值 为 some_value 的 
$_GET['some var'] 变 量 , 值 为 20 的 $_GET['age'] 变 量 ， 等 等 。 

正如 本 书 介绍 的 那样 ， 可 以 通过 创建 一 个 使 用 GET 方 法 的 HTML 表 单 来 传递 数据 。 使 用 这 种 
方法 ， 也 可 以 向 PHP 页 面 发 送 数据 ， 而 无 需 使 用 表单 。 通 常情 况 下 ， 可 以 通过 在 另 一 页 中 使 用 链 
接 的 方法 做 到 

<a href="page.php?id=22">Some Link</a> 

在 从 数据 库 中 读 取 数 据 后 ， 这 个 链接 能 够 由 PHP 动 态 生 成 ， 它 将 向 page .php 传 递 值 22 并 可 
访问 $_GET['id']。 

下 面 两 段 代码 将 通过 使 用 一 个 硬 编码 的 HTML 页 面 诠释 这 个 概念 ， 请 自行 进行 尝试 。 


这 创建 HTML 页 面 
































(GD 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 文档 ， 命 名 为 hello .html (参看 脚本 3-6): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Greetings!</title> 

</head> 

<body> 

<1-- 脚本 3-6 - hello.html --> 

<div><p>Click a link to say hello:</p> 

















脚本 3-6 ”这 个 HTML 页 面 使 用 链接 来 向 PHP 脚 本 传递 值 (由 此 效仿 使 用 GET 方 法 的 表单 ) 


1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Greetings!</title> 

到 </head> 

8 <body> 

9 <!-- 脚本 3-6 - hello.html --> 

10 <div><p>Click a link to say hello:</p> 


11 

12. -各 工交 

13 <li><a href="hello.php?name=Michael">Michael</a></1i> 
14 <li><a href="hello.php?name=Celia">Celia</a></1i> 

15 <li><a href="hello.php?name=Jude">Jude</a></1i> 


16 <li><a href="hello.php?name=Sophie">Sophie</a></1i> 
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17 </ul> 


19 </div> 
20 </body> 
21 </html> 


(2) 创建 一 个 链接 指向 PHP 肢 本， 通过 下 面 的 URL 传 递 值 : 


<ul> 
<li><a href="hello.php?name=Michael">Michael</a></1i> 
<li><a href="hello.php?name=Celia">Celia</a></1i> 
<li><a href="hello.php?name=Jude">Jude</a></1i> 
<li><a href="hello.php?name=Sophie">Sophie</a></1i> 
</ul> 


这 样 做 的 前 提 是 用 户 看 到 一 个 链接 的 列表 ， 每 个 链接 都 同一 个 特定 的 名 称 关联 (参见 图 
3-14)。 当 用 户 点 击 链 接 时 ， 将 会 在 URL 中 把 与 名 称 对 应 的 值 发 送 至 hello .php (参见 图 3-15)。 











Click a link to Say hello: 


。 Michael 
。 Celia 
es。 Jude 


Sophie 























图 3-14 ”这 个 简单 的 HTML 页面 拥有 4 个 指向 PHP 脚 本 的 链接 




















<!1 -~ Script 3.6 - hello.html ~--> 

<div><p>Click a link to say hello:</p> 

<ul> 
<li><a href="hello.php?name=Michael">Michael</a></1i> 
<li><a href="hello.php?name=Celia">Celia</a></li> 
<li><a href="hello.php?name=Jude">Jude</a></1i> 
<li><a href= "hello.php?name=Sophie">Sophie</a></1i> 

</ul> 

</div> 























图 3-15 ”HTML 源 代码 显示 出 这 些 值 是 如 何 通 过 4 个 链接 进行 发 送 的 


如 果 希 望 使 用 不 同 的 名 称 ， 这 样 很 好 ,但 是 请 坚持 使 用 只 有 一 个 单词 的 名 称 形式 ， 而 不 
要 在 中 间 加 上 空格 或 者 标点 符号 (否则 它们 不 能 被 正确 地 传送 至 PHP 肢 本， 我 们 将 对 此 进行 





























说 明 )。 
(3) 完成 HTML 页 面 : 
</div> 
</body> 
</html> 


(4) 将 脚本 保存 为 hello .html 并 且 在 启用 了 PHP 的 服务 器 上 保存 至 适当 的 目录 。 

(5) 在 Web 浏 览 器 中 通过 URL 加 载 这 个 HTML 页 面 。 

可 以 通过 点 击 页 面 上 的 链接 访问 PHP 脚 本 ,这样 就 能 不 用 输入 URL 而 查看 到 HTML 页 面 ,但 
是 在 这 里 请 从 使 用 URL 开 始 。 先 不 要 点 击 任何 链接 ， 因 为 PHP 脚 本 并 不 存在 。 
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过 创建 PHP 脚 本 








(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 文档 ， 命 名 为 hello .php (参看 脚本 3-7): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Greetings!</title> 

</head> 

<body> 


脚本 3-7 这 个 PHP 页 面 在 URL 中 引用 了 name 的 值 ， 用 来 打印 一 个 问候 

















和 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Greetings!</title> 

冰 </head> 

8 <body> 

9 <?php // 脚本 3-7 - hello.php 


ini_set ('display_errors'，1); // 让 我 从 错误 中 学 到 知识 ! 
error_reporting (E_ALL | 已 STRICT); // 显示 所 有 可 能 出 现 的 问题 ! 


// 该 页 面 将 从 URL 中 接收 name 的 值 。 
// 打印 Hello: 


17 S$name = $_GET['name']; 
18 print "<p>Hello, <span style=\"font-weight: bold;\">$name </span>!</p>"; 





20 ?> 
21 </body> 
22 </html> 


(2) 开始 编写 PHP 代 码 . 
<?php // 脚本 3-7 - hello.php 
(3) 如 果 需 要 ， 对 错误 管理 进行 说 明 : 


ini_set ('display_errors', 1); 


error_reporting (E_ALL E_STRICT); 


行 代码 对 PHP 如 何 进行 错误 响应 做 了 设置 ， 这 在 本 节 之 前 已 经 做 过 介绍 了 。 它 们 不 是 必 
须 的 ， 。 有 所 帮助 。 
(4) 在 URL 中 传递 name 的 值 来 创建 一 个 问候 : 


Sname = $_GET['name']; 
print "<p>Hello, <span style= \"font-weight: bold;\">$name </span>!</p>"; 
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通过 URL 将 name 变 量 传送 给 页 面 (参看 脚本 3-6)。 引 用 $_GET[ 'name'] 以 访问 这 个 值 。 可 
以 使 用 $_GET ( 同 $_POST 相 对 )， 这 是 因为 值 是 通过 GET 请 求 传 送 的 。 

如 同 之 前 的 PHP 脚 本 一 样 ， 预 定义 变量 ($_GET) 中 的 值 被 首先 赋 给 另外 一 个 变量 ， 以 简化 
在 print 语 句 中 的 语法 。 

(5) 完成 PHP 代 码 和 HTML 页 面 : 


?> 
</body> 
</html> 























(6) 将 脚本 保存 为 hello.php 并 且 在 启用 了 PHP 的 服务 器 上 保存 到 适当 的 目录 。 它 应 该 与 
hello.html (脚本 3-6) 保存 在 相同 的 目录 下 。 
(0) 点 击 hello .html 中 的 链接 ， 查 看 结果 (参见 图 3-16 和 图 3-17)。 


QQNN Greetings! 
||P + |Qyhttp://phpvqs4:8888/hello.php?rname=Michael 



































Hello, Michael! 












































图 3-16 ”通过 点 击 第 一 个 链接 ，Michael 作 为 name 的 值 在 URL 中 进行 传送 ， 最 终 出 现在 问候 中 
AQNAN Greetings! 
[<1»] [+ |@D http://phpvqs4:8888/hello.php?rname=Celia 
Hello, Celia! 
图 3-17 通过 点 击 最 后 一 个 链接 ，Celia 作 为 name 的 值 在 URL 中 进行 传送 ， 最 终 出 现在 问候 中 
w 提示 


口 如 果 直 接 运行 hel1o .php， 将 获得 一 个 错误 通知 ， 这 是 因为 没有 name 的 值 将 通过 URL 进 
行 传送 (参见 图 3-18)。 








@NN Greetings! 


[<|»| [+ [http://phpvas4:8888/hello.php C | 








Notice: Undefined index: name in 
/Users/larryullman/Sites/phpvqs4/hello.php on line 17 





Hello, ! 














图 3-18 ”如 果 $_GET['name' 1] 变量 没有 被 赋值 ， 浏 览 器 将 打印 出 这 样 带 有 错误 通知 的 难堪 信息 





口 由 于 hel1o .php 通 过 URL 读 取 值 ， 因 此 它 独 立 于 hel1lo .html 运 行 。 例如， 可 以 直接 编辑 
hello.php 中 的 URL 来 使 用 name 的 值 进行 问候 ， 即 使 hello.html 并 不 包含 指向 那样 
name 的 链接 (参见 图 3-19)。 
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QO0 Greetings! 


[+| (yhttp://phpvqs4:8888/hello.php?name=Nicole 


Hello, Nicole! 

















图 3-19 ”在 URL 中 赋 给 name (小 写 ) 的 任何 值 ， 都 会 被 PHP 脚 本 问候 

















口 如 果 希 望 使 用 链接 向 脚本 传送 多 个 值 ， 将 variable=value 对 (例如 ，first_name = 
Larry) 用 与 号 (&) 进行 分 隔 。 因 此 这 里 的 链接 将 是 hello.php? first_name=Larry& 
last_name=Ullman。 在 学 习 urlencode() 函数 (后面 会 介绍 ) 之 前 ， 最 好 继续 使 用 单 
个 单词 ， 不 要 添加 标点 或 空格 。 

口 虽然 这 里 的 示例 为 个 人 的 name 设 置 了 值 ， 这 也 许 并 不 实际 ， 但 是 这 种 基础 技术 在 很 多 场 
景 中 却 非 常 有 用 。 例 如 ， 一 个 PHP 脚 本 可 以 构成 一 个 模板 ， 根据 页 面 从 URL 中 接收 到 的 值 
的 不 同 ， 产 生 具 有 不 同 内 容 的 Web 页 面 。 


3.7 ”回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
com/forum/。 











3.7.1 回顾 


口 表单 中 action 属 性 的 作用 是 什么 ? 

口 表单 中 method 属 性 的 作用 是 什么 ? GET 与 POST 哪 个 更 安全 ?哪个 方法 可 以 在 浏览 器 中 
添加 为 书签 。 

口 哪些 预定 义 变量 包括 从 表单 提交 的 数据 ? 注意 : 答案 不 止 一 个 。 

口 某 个 HTML 页 面包 括 提 交 到 PHP 脚 本 的 表单 ， 为 什么 必须 通过 URL 加 载 该 HTML 页面 。 

口 在 哪 种 情况 下 ， 在 脚本 中 尝试 启用 display_errors 会 失败 ?为 什么 在 实际 网 站 上 启用 


display_errors 会 不 安全 ? 

















3.7.2 ”实践 


口 不 通过 URL 在 Web 浏 览 器 中 加 载 feedback.html (如 在 地 址 栏 中 以 file:/ 开 头 填 地 址 ) , 填 
写 表单 后 提交 。 仔 细 观 察 结 果 。 你 应 该 能 够 识别 问题 ， 并 了 解 问题 的 原因 ， 这 样 今后 出 
现 类 似 问 题 时 ， 你 就 可 以 解决 了 。 

口 确保 在 你 的 开发 环境 中 启用 display_errors。 

口 确保 在 你 的 开发 环境 中 error_reporting 已 经 设置 为 E_ALL |E_STRICT。 

口 尝试 在 一 个 PHP 脚 本 中 引入 不 同 的 问题 (例如 ，3 引 | 号 不 对 称 、 漏 掉 分 号 、 错 误 地 引用 变量 
等 )， 再 观察 结果 。 
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口 做 一 个 实验 ， 向 hello .html 和 hello .php 页 面 传 入 不 同 的 值 (包括 数字 )， 通 过 URL 提 
交 到 PHP 脚 本 。 

口 在 hello.htm1l 上 创建 一 个 变量 ， 发 送 多 个 name=value 对 到 PHP 脚 本 。PHP 脚 本 是 否 打 
印 所 有 接收 到 的 值 ? 

口 如 果 你 想 知道 能 够 传人 的 值 类 型 ， 又 不 介意 等 竺 的话， 可 以 试 着 通过 URL 向 页 面 传人 一 
些 复杂 的 值 (如 空格 或 标点 ) ， 看 看 会 发 生 什么 。 

口 创建 一 个 新 的 HTML 表 单 ， 用 它 执行 一 个 假想 的 任务 (或 是 一 个 更 简单 的 任务 ) 。 创 建 PHP 
脚本 处 理 该 表单 ， 打 印 收 到 的 数据 。 




















使 用 数值 








本 章 内 容 

口 创建 表单 

口 执行 算术 运算 

口 格式 化 数值 

口 理解 优先 级 

口 数值 的 自 增 和 自 减 
口 创建 随机 数 

口 回顾 和 实践 





在 第 2 章 中 ， 我 们 讨论 了 变量 的 不 同类 型 ， 如 何 为 它们 赋值 以 及 在 通常 情况 下 变量 的 用 法 。 
我 们 将 专门 在 本 章 介绍 两 种 数值 型 变量 一 一 整 型 (整数 ) 和 浮 点 数值 (又 称 为 浮 点 或 小 数 )。 

本 章 首先 讲述 创建 HIML 表 单 ， 它 用 来 创建 一 些 数值 型 变量 。 然 后 将 开始 学 习 如 何 执行 基本 
算术 操作 、 如 何 对 数值 进行 格式 化 ,以 及 如 何 处 理 运 算 符 优先 级 。 本 章 的 最 后 两 节 将 涉及 自 增 和 
自 减 数值 以 及 生成 随机 数 。 本 章 通 篇 都 将 讨论 与 数值 相关 的 PHP 实 用 函数 。 


4.1 创建 表单 


本 章 中 大 多 数 PHP 示 例 都 将 在 电子 商务 前 提 下 执行 各 种 计算 。 一 个 表单 可 能 会 接受 价格 、 数 
量 、 折 扣 值 、 税 率 以 及 运输 费 (参见 图 4-1)， 并 且 处 理 表 单 的 PHP 脚 本 将 会 返回 总 体 费 用 。 这 个 
费用 将 会 以 用 户 希 望 分 期 付款 的 期 数 进行 拆 解 ， 从 而 生成 月 费用 的 值 (参见 图 4-2)。 


Fill out this form to calculate the total cost: 
































Price: |5.00 

Quantity: 100 

Discount: 10.00 

Tax: 75 |(%) 

Shipping method: | Slow and steady ™ 


Number of payments to make: 10 


Calculatel 























图 4-1 ”这 个 表单 接受 从 用 户 获取 的 数值 ， 并 把 它们 发 送 给 PHP 页 面 
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You have selected te purchase: 

100 widgetfs) at 

$5.00 price each plus a 

$5.00 shipping cost and a 

TS percent ta rate. 

Lfter your $10.00 discount, the total cost is $532.13. 

Divided over 10 monthly payments, that would be $53.21 each. 


图 4-2 PHP 脚 本 会 根据 用 户 提 交 的 数据 计算 费用 并 返回 结 
果 。 这 是 本 章 结束 时 此 表单 的 运算 结果 


让 我 们 从 创建 一 个 HTML 页 面 开始 学 习 的 历程 ， 用 户 可 以 通过 这 个 页 面 输入 不 同 的 值 。 


之 创建 HTML 表 单 的 步骤 cE 


(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 HTML 文 档 ， 命 名 为 calculator.html (参看 脚本 
























































<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Product Cost Calculator </title> 

</head> 

<body><!-- 脚本 4-1 - calculator.html --> 

<div><p>Fill out this form to calculate the total cost:</p> 





脚本 4-1 这 个 基本 的 HTML 表 单 创建 了 将 在 PHP 脚 本 中 执行 算术 计算 的 数值 











1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

E! <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Product Cost Calculator</title> 

7 </head> 

8 <body><!-- 脚本 4-1 - calculator.html --> 

9 <div><p>Fill out this form to calculate the total cost:</p> 

10 

11 <form action="handle calc.php" method="post"> 

训 

13 <p>Price: <input type="text" name="price" size="5" /></p> 

14 

15 <p>Quantity: <input type="text" name="quantity" size="5" /></p> 
16 

17 <p>Discount: <input type="text" name="discount" size="5" /></p> 
18 

19 <p>Tax: <input type="text" name="tax" size="3" /> (%)</p> 

20 

21 <p>Shipping method: <select name="shipping"> 

22 <option value="5.00">Slow and steady </option> 

23 <option value="8.95">Put a move on it. </option> 

24 <option value="19.36">I need it yesterday!</option> 


0 040 0000 





25 </select></p> 


26 

27 <p>Number of payments to make: <input type="text" name="payments" size="3" /></p> 
28 

29 <input type="submit" name="submit" value="Calculate!" /> 
30 

31 </form> 

32 

33 </div> 

34 </body> 

357 </hitml> 

(2) 创建 初始 form 标 签 





<form action="handle calc.php" method="post"> 


个 form 标 签 标志 着 HTML 表 单 的 开始 。 它 的 action 属 性 指明 表单 数据 将 被 提交 给 名 为 
ee 该 标签 的 method 属 性 告诉 页 面 使 用 POST 方 式 来 传送 数据 。 参 看 第 3 
章 以 获取 更 多 的 有 关 信 息 。 

(3) 为 price、quantity、discount 和 tax 创 建文 本 框 : 


<p>Price: <input type="text" name="price" size="5" /></p> 
<p>Quantity: <input type="text" name="quantity" size="5" /></p> 
<p>Discount: <input type="text" name="discount" size="5" /></p> 
<p>Tax: <input type="text" name="tax" size="3" /> (%)</p> 


HTML 没 有 给 数值 提供 输入 类 型 ， 因 此 需要 创建 文本 框 以 供 值 的 输入 。 插 入 的 说 明 指明 tax 
的 格式 是 百分数 。 

请 记 住 , 文本 输入 框 使 用 的 name 命 名 方式 应 当 同 有 效 的 PHP 变 量 名 命名 方式 相符 〈 只 能 是 字 
母 、 数 字 和 下 划 线 ， 并 且 不 能 以 数字 开头 ， 等 等 ) 。 

(4) 添加 一 个 字段 供用 户 选择 shipping 的 方式 ， 


<p>Shipping method: <select name="shipping"> 
<option value="5.00">Slow and steady</option> 
<option value="8.95">Put a move on it.</option> 
<option value="19.36">I need it yesterday!</option> 
</select></p> 


这 个 shipping 选 项 是 用 下 拉 菜 单 实现 的 。 每 个 选项 的 值 就 是 选择 该 选项 所 对 应 的 shipping 
方式 产生 的 费用 。 因 此 ， 如 果 用 户 选 择 诸如 Put a move on it 选 项 时 ，$_POST['shipping'] 在 
handle_calc.php 中 获取 到 的 值 将 是 8.95。 

(5) 完成 HTML 表 单 : 

<p>Number of payments to make: <input type="text" name="payments" 

一 Size="3" /></p> 


<input type="submit" name="submit" value="Calculate!" /> 
</form> 


最 后 两 个 输入 区 域 接 受 了 分 期 付款 所 需 的 期 数 ， 并 且 创 建 了 一 个 提交 按钮 (标签 为 
Calculate!)。 关 闭 form 标 签 标志 着 页 面 中 表单 部 分 的 结束 。 
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(6) 完成 HTML 页 面 。 


</div> 
</body> 
</html> 


(7) 将 脚本 保存 为 calculator .html 并 在 Web 浏 览 器 上 查看 。 
由 于 这 是 一 个 HTML 页 面 ， 因 此 可 以 在 Web 浏 览 器 上 直接 查看 。 


4.2 ”算术 运算 
ia a 基础 数学 涉及 加 、 减 、 乘 、 除 的 法 则 。 
使 用 最 显而易见 的 运算 符 表示 : 
口 加 (+); 
口 减 (一 ); 
口 乘 (*); 
口 除 (/)。 











演示 上 述 运 算 符 的 使 用 方法 ,我 们 将 创建 一 个 PHP 脚 本 来 计算 一 些小 商品 售卖 
以 计算 的 脚本 将 是 一 个 基本 的 购物 车 应 用 程序 一 一 一 种 非常 实用 的 Web 页 面 特性 (但 在 这 


中 ， 相 关 数 值 变量 的 值 将 来 自 于 calculator .html)。 


这 些 运算 法 则 都 在 PHP 中 





的 总 费用 。 用 


文 个 示例 


编写 这 个 脚本 时 ， 请 务必 注意 注释 的 用 法 〈 参 看 脚本 4-2) ， 它 在 代码 后 阐明 了 不 同 代码 行 的 





意义 以 及 生成 的 原因 。 
脚本 4-2 ”PHP 脚本 用 表单 提交 的 数值 进行 了 所 有 的 标准 算术 计算 











19 S$price = $_ POST['price']; 

20 Squantity = $_ POST['quantity']; 
21 S$discount = $_POST['discount']; 
22 S$tax = $_ POST['tax']; 

23 $shipping = $_POST['shipping']; 





charset=utf-8"/> 


1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
3 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; 

6 <title>Product Cost Calculator</title> 

2 <style type="text/css" media="screen"> 

8 .number { font-weight: bold; } 

9 </style> 

10 </head> 

11 <body> 

12 <?php // 脚本 4-2 - handle_calc.php 

13 /* 脚本 calculator.html 获 取 数 值 

14 并 计算 总 成 本 和 月 付款 数 。*/ 

15 

16 // 可 以 在 此 进行 错误 处 理 。 

17 

18 // 从 $_POST 数 组 获取 数值 : 








64 U40 0000 
24 S$payments = $_ POST['payments']; 
25 
26 // 计算 总 额 : 
27 S$total = $price * S$quantity; 
28 S$total = $total + $shipping; 
29 S$total = $total - $discount; 
30 


31 // 定义 税率 : 
32 S$taxrate = $tax/100; 


33 S$taxrate = S$taxrate + 1; 


34 
35 // 纳入 税率 后 的 总 成 本 : 


36 S$total = $total * $Laxrate7 


37 
38 // 计算 月 付款 数 : 


39 $monthly = S$total / S$payments; 


40 
41 // 打印 结果 : 


42 print "<p>You have selected to purchase:<br/> 
43 <xspan class=\"number\">$quantity </span> widget(s) at <br/> 


44 $<span class=\"number\ 


">$price</span> 


price each plus a <br/> 


45 $<span class=\"number\ 


">$shipping </span> shipping cost and a <br/> 


46 <span class=\"number\">$tax</span> percent 七 ax rate.<br/> 
47 After your $<span class=\"number\"> $discount</span> discount, the total 


cost is 
48 $<span class=\"number\ 


">$total </span>.<br/> 


49 Divided over <span class=\"number\"> $payments</span> monthly payments, 
that would be $<span class=\"number\"> $monthly</span> each.</p>"; 


50 

日 上 ， 汪 交 

52 </body> 
S53 .CmL> 


> 创建 销售 费用 计算 器 








(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 文档 ， 命 名 为 handle_calc .php (参看 脚本 4-2) : 





<!IDOCTYPE html PUBLIC "-// 








W3C//DTD XHTML 1.0 Transitional//EN" 


"http://www.w3.org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 


<html xmlns="http://www.w3 
<head> 


.Org/1999/xhtml" xml:lang="en" lang="en"> 





<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 


<title>Product Cost Calc 

<style type="text/css" 

.number { font-weight: 
</style> 

</head> 

<body> 





ulator </title> 
media="screen"> 
bold;} 


在 脚本 的 开头 处 定义 了 一 个 名 为 number 的 CSS 类 。 页 面 中 任何 带 有 这 个 类 值 的 元 素 都 将 被 
加 粗 。 换 句 话 说 , 当 脚 本 打印 输出 时 , 表单 中 的 输入 数值 和 各 种 计算 结果 都 会 以 更 明显 的 方式 ( 粗 


体 ) 显示 出 来 。 
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(2) 如 果 需 要 ， 可 以 插入 PHP 标 签 和 地 址 错误 处 理 ; 

<?php // 脚本 4-2 - handle_calc.php 

可 以 依照 当前 PHP 的 设置 情况 ,添加 一 些 代码 行 来 开启 display_errors， 并 调整 错误 报告 
的 级 别 。 参 看 第 3 章 中 的 相关 信息 。 

(3) 将 $_PosT 元 素 的 值 赋值 给 局 部 变量 : 


Sprice = $_POST['price']; 
$sSquantity = $_POST['quantity']; 
Sdiscount = $_POST['discount']; 
Stax = $_POST['tax']; 

sshipping = $_POST['shipping']; 
Spayments = $_POST['payments']; 


脚本 将 获取 到 在 预定 义 的 $_PoST 变 量 中 所 有 的 表单 数据 。 为 了 访问 单个 的 表单 值 ， 引 用 
$_POST['index'] 以 将 index 赫 换 为 相应 表单 元 素 的 name 值 。 这 些 值 在 这 里 被 赋 给 单个 的 局 部 
变量 ,以便 在 脚本 剩余 部 分 更 加 方便 地 使 用 它们 。 

请 注意 ， 每 个 变量 都 被 给 予 了 一 个 描述 性 名 称 ， 并 且 全 部 使 用 小 写字 母 命名 。 

(4) 开始 计算 总 费用 : 


Stotal = S$price * S$quantity; 
Stotal = Stotal + $shipping; 
Sotal = S$total - S$discount; 


星 号 (*) 在 PHP 中 代表 乘法 ， 因 此 total 首 先 被 计算 为 购买 的 数量 ($quantity) 乘 以 单 
价 ($price)。 运 输 费 ($shipping) 需要 添加 进 总 费用 中 (请 记 住 ， 运 输 费 与 shipping 下 拉 
菜单 中 所 选择 项 的 value 属 性 相关 )， 并 且 减 去 折扣 ($discount)。 

注意 在 为 变量 赋值 的 时 候 ， 可 以 在 其 中 使 用 变量 的 现 有 值 ， 这 是 完全 可 以 接受 的 (正如 在 最 
后 两 行 中 所 做 的 那样 )。 

(5) 计算 税率 和 新 的 总 费用 : 

Staxrate = S$tax/100; 


staxrate = S$taxrate + 1; 
Stotal = Stotal * S$taxrate; 


税率 ($tax) 应 当 是 一 个 百分比 ， 例 如 将 8 或 者 5.75 除 以 100， 转 化 为 同 百分比 等 同 的 小 数 
《0.08 或 者 0.0575) 。 最 后 将 这 个 结果 加 上 1 计算 购买 物品 的 实际 税率 ， 然 后 乘 以 先前 计算 的 总 价 。 
这 在 算术 上 相当 于 用 小 数 类 型 的 税率 ($taxrate) 乘 以 总 费用 ($total)， 得 到 的 结果 成 为 总 
费用 ($total) 的 新 值 〈 例 如 ， 消 费 100 美 元 需 缴纳 5% 的 税 ， 因 此 总 共 将 花费 105 美 元 ， 这 就 相 
当 于 用 100 美 元 直接 乘 以 1.05 得 到 的 结果 )。 

(6) 计算 月 付费 用 : 

Smonthly = Stotal / S$payments; 

作为 除法 的 示例 , 我 们 假设 购买 某 物 或 者 任何 需要 分 期 付款 的 购买 行为 将 会 持续 数 月 。 因 此 ， 
我 们 需要 将 总 费用 除 以 需要 分 期 付款 的 月 数 以 获得 月 付费 用 。 

(7) 打印 结果 : 
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print "<p>You have selected to purchase:<br/> 
<span class=\"number\">$quantity </span> widget(s) at <br/> 
$<span class=\"number\">$price </span> price each plus a <br/> 
$<span class=\"number\">$shipping </span> shipping cost and a <br/> 
<span class=\"number\">$tax</span> percent tax rate.<br/> 
After your $<span class=\"number\"> $discount</span> discount, the total cost is 
$<span class=\"number\">s$total </span>.<br/> 
Divided over <span class=\"number\"> S$payments</span> monthly payments, 
”that would be $<span class=\"number\">s$monthly</span> each.</p>"; 


print 语 句 将 每 个 值 连同 一 些 文本 一 起 发 送 到 Web 浏 览 器 。 为 了 增加 可 读 性 ， 增 加 <br/> 标 
签 以 格式 化 浏览 器 的 显示 结果 。 另 外 ，pzrint 国 数 将 结果 处 理 为 多 行 显示 ， 以 让 PHP 代 码 看 上 去 
更 加 清晰 。 每 个 变量 的 值 在 浏览 器 中 都 被 突出 显示 ， 使 用 的 方法 是 用 带 有 number 类 属性 的 Span 
标记 对 它们 进行 折 行 。 

(8) 关闭 PHP 代 码 段 并 且 完 成 HIML 页面。 





























OO 

</html> 

(9) 将 脚本 保存 为 handle_calc .php， 并 放 在 启用 了 PHP Fi out this form to calculate the total cost 
的 服务 器 上 适当 的 目录 下 。 Price: [6 

确保 同 calculator .html 在 同一 目录 下 。 Quantity: [6 

(10) 在 Web 浏 览 器 中 测试 脚本 (参见 图 4-3 和 图 4-4)。 人 





本 书 不 想 在 这 点 上 做 过 多 宛 长 的 解释 , 但 是 请 确保 你 确实 
是 通过 URL (http://something) 来 加 载 HTML 表 单 的 ， 这 样 当 
它 被 提交 时 ，PHP 脚 本 可 以 通过 这 个 URL 运 行 。 

可 以 用 这 些 值 进行 试验 , 验证 计算 器 是 否 正常 运行 。 如 果 
漏 掉 了 一 些 值 , 结果 将 显得 有 一 些 怪异 ,但 是 计算 仍 将 继续 进 。。 四 表征 如 图 显 不 
行 (参见 图 4.5)。 


Shipping method: | Slow and steady ~ 

















Number of payments to make: 12 


Calculatel 
























































You have selected to purchase 

6 wdeetls) at 

$19.95 price each plus a 

$5.00 shipping cost and a 

6 percent tas rate 

LHer your $10.00 discount, the total cost is $121.582 

Divided over 12 monthly payments, that would be $10.1319333333 cach. 


图 4-4 计算 结果 




















You hawve selected to Purchase 

6 wideetts) at 

$19.95 price each plys a 

$5.00 shipping cost and a 

percent tax rate. 

Bfter your $ discount, the total cost is $124.7. 

Divided over 12 monthly payrments, that would be $10.3916666667 each 














器 








4-5 可 以 漏 掉 或 者 更 改 任何 值 然 后 重新 运行 计算 器 。 这 里 将 漏 掉 ftax 和 aiscount 的 值 
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w 提示 

口 你 一 定 会 注意 到 , 计算 器 计算 出 来 的 结果 同 真实 的 美元 值 并 不 相符 (参见 图 4-4 和 图 4-5)。 

4.3 市 将 介绍 如 何 处 理 这 样 的 结果 。 

口 如 果 希 望 在 税 值 或 者 折扣 值 (或 两 者 都 ) 前 打印 总 费用 的 值 , 可 以 使 用 两 种 方法 ,在 $total 
变量 的 值 被 再 次 变更 之 前 , 在 合适 的 值 后 插入 适当 的 print 语 句 。 或 者 使 用 新 变量 存储 后 
来 计算 结果 的 值 (例如 ，sStotal_with tax 和 S$total_less_discount)。 

口 由 于 变量 以 美元 符号 开头 ， 因 此 在 打印 带 有 美元 符号 的 数字 时 (比如 $10，10 通 过 变量 产 

生 ) 需要 格外 小 心 。 不 能 使 用 $$variable 变 量 , 这 是 因为 两 个 美元 符号 的 组 合 将 产生 一 

个 非常 复杂 的 变量 类 型 ， 而 本 书 将 不 对 其 进行 讨论 。 解 决 方案 是 在 美元 符号 中 间 放 置 一 

些 东 西 。 例 如 ， 在 本 示例 中 ， 美 元 符号 和 变量 名 之 间 被 放置 了 一 个 空格 或 者 HTML 标 签 。 

另 一 个 选择 是 将 第 一 个 美元 符号 进行 转 义 : 

print "The total is NSStotal" ， 

第 三 种 选择 是 使 用 连接 字符 ， 这 会 在 第 5 章 介绍 。 

这 个 脚本 会 根据 提交 字段 的 不 同 显 示 不 同 的 结果 。 唯 一 可 能 产生 问题 的 字段 是 月 付 的 数 

量 : 如 果 它 被 漏 掉 ， 将 会 看 到 除 以 零 的 警告 。 第 6 章 将 讨论 在 使 用 表单 数据 前 对 它们 的 有 

效 性 进行 验证 。 

口 HIML 5 中 预计 会 有 一 个 或 多 个 限制 用 户 输入 数值 的 文本 框 。 


4.3 ”格式 化 数值 


尽管 已 经 可 以 使 用 计算 器 , 但 是 现在 仍然 存在 一 个 合理 性 的 问题 : 月 付 金额 不 能 是 10.13183333 
美元 的 形式 。 为 了 生成 更 多 可 用 的 数值 ， 需 要 对 它们 进行 格式 化 。 

有 两 个 国 数 适 合 达到 格式 化 数值 的 目的 。 第 一 个 是 zound() ， 它 用 于 对 数值 截取 特定 位 数 的 
小 数 。 函 数 的 第 一 个 参数 是 需要 格式 化 的 数值 ， 它 既 可 以 是 一 个 数字 ,也 可 以 是 一 个 存 有 数值 的 
变量 。 第 二 个 参数 是 可 选 的 ， 它 代表 需要 取 的 小 数位 数 。 例 如 : 

round (4.30); // 4 

round (4.289, 2); // 4.29 

Snum = 236.26985; 

round ($num); // 236 

另外 一 个 可 以 用 来 格式 化 数值 的 函数 是 number_format ()。number_format() 同 round() 
的 工作 方式 类 似 , 它 有 一 个 数值 参数 (或 者 有 数字 值 的 变量 ) 以 及 一 个 可 选 的 小 数位 数 指定 参数 。 
这 个 函数 通过 千 位 分 组 来 格式 化 数字 ， 格 式 化 的 效果 通常 如 下 所 示 : 

number_format (428.4959, 2); // 428.50 


number_format (428, 2); // 428.00 
number_format (123456789); // 123,456,789 


让 我 们 将 数值 进行 适当 的 格式 化 并 重新 编写 PHP 脚 本 。 



































口 
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之 格式 化 数值 的 步 又 








(1) 在 文本 编辑 器 或 IDE 中 打开 handle_calc.php (参看 脚本 4-2)。 
@2) 在 所 有 计算 代码 之 后 、print 语 句 之 前 ， 添 加 下 面 的 代码 (参看 脚本 4-3 ) : 


Stotal = number_format ($total, 2); 
smonthly = number_ format ($monthly, 2); 


脚本 4-3 


db 
2 
3 
4 
5 
6 
涉 
8 
9 





be he hs ES 
Wy OO OO Wb WP 


WMD 2 
OOO 


DD 
\D oo 


LO LO W W W W 
从中 心口 


LO LU CO 
\D 0o 局 


心 心心 心 
WwW N 上品 





对 数值 变量 的 值 应 用 number_format () 函数 ， 以 让 它们 符合 实际 使 用 的 要 求 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Product Cost Calculator</title> 
<style type="text/css" media="screen"> 

.number { font-weight: bold;} 
</style> 

</head> 

<body> 

<?php // 脚本 4-3 - handle _ calc.php #2 

/* 脚本 从 calculator.html 获 取 数 值 

并 计算 总 成 本 和 月 付款 数 。*/ 





// 可 以 在 此 进行 错误 处 理 。 


// 从 $_POST 数 组 获取 数值 : 

$Sprice = $_POST['price']; 
squantity = $_POST['gquantity']; 
$discount = $_POST['discount']; 
Stax = $_POST['tax']; 

$shipping = $_POST['shipping']; 
Spayments = $_POST['payments']; 





// 计算 总 额 : 

Stotal = S$price * Squantity; 
Stotal = S$total + $shipping; 
Stotal = Stotal - $discount; 


// 定义 税率 : 
$staxrate = $tax/100; 
Staxrate = S$taxrate + 1; 


// 纳入 税率 后 的 总 成 本 : 
$Stotal = S$total * S$taxrate; 


// 计算 月 付款 数 : 
smonthly = Stotal / S$payments; 


// 格式 化 数值 ， 保 留 2 位 小 数 。 
$total = number format ($total, 2); 
$smonthly = number format ($monthly, 2); 
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44 

45 // 打印 结果 : 

46 print "<p>You have selected to purchase:<br/> 

47 <span class=\"number\">$quantity</span> widget(s) at <br/> 

48 $<span class=\"number\">$price</span> price each Plus a <br/> 

49 $<span class=\"number\">$shipping</span> shipping cost and a <br/> 

50 <span class=\"number\">$tax</span> percent tax rate.<br/> 

51 After your $<span class=\"number\"> $discount</span> discount, the total cost is 

52 $<span class=\"number\">$total </span>.<br/> 

53 Divided over <span class=\"number\"> S$payments</span> monthly payments, that 
would be $<span class=\"number\">$smonthly </span> each.</p>"; 


54 
55- RS 
56 </body> 





57 </html> 

格式 化 这 些 数值 ， 可 以 在 每 个 计算 操作 完成 之 后 应 用 这 些 函 数 , 但 是 一 定 要 在 它们 被 传送 给 
Web 浏 览 器 之 前 。 第 二 个 参数 (2) 指明 结果 数值 需要 保留 2 位 小 数 , 这 个 设置 会 对 数值 四 舍 五 入 ， 
并 且 会 在 需要 的 时 候 用 零 补 位 。 

(3) 将 文件 保存 在 同 calculator .html 相 同 的 目录 下 ， 并 在 浏览 器 上 测试 (参见 图 4-6 和 图 
4-7)。 

















Fill out this form to calculate the total cost: 
Price: 199.00 
Quantity: 4 


Discount: 125.00 





You hawve selected to purchase: 
Tax: 55 ob) 4 wideetis) at 
$99.00 price each plus a 


Shippl thod: | Put it i 
PP $3.95 shipping cost and a 






























































Number of payments to make: 124 与 .SS percent tax rate. 
Bfter your $25.00 discount, the total cost is $400.85. 
Divided over 24 monthly payments, that would be $16.70 each. 
图 4-6 “另外 一 次 执行 表单 的 图 4-7 ”脚本 的 更 新 版 本 返回 了 更 多 符合 实际 的 值 ， 这 
结果 得 益 于 number_format () 国 数 处 理 的 结果 


w 提示 

口 此 外 , 还 有 很 多 更 加 复杂 的 方式 都 使 用 printf() 和 sprintf() 函数 来 格式 化 数值 。 由 于 

它们 的 语法 比较 复杂 ， 因 此 本 书 将 不 进行 讨论 。 请 参看 PHP 手 册 以 获得 更 多 的 相关 信息 。 

口 非 Windows 版 本 的 PHP 有 money_format () 国 数 , 它 可 以 用 在 有 number_format () 场合 。 

口 出 于 一 些 复杂 的 原因 ，zrouna ( ) 函数 对 于 “精确 ”半数 的 情况 〈0.5、0.05、0.005 等 )， 合 
去 和 进 上 的 次 数 是 一 半 对 一 半 。 


























@ 这 里 的 意思 是 如 果 需 要 round ( ) 函数 处 理 的 数字 其 小 数 部 分 正好 是 一 半 ， 则 取 整 到 最 接近 的 偶数 ， 并 遵守 “四 使 
六 入 五 成 双 ” 原 则 ， 即 前 一 位 是 奇数 ， 则 进 1， 前 一 位 是 偶数 则 舍 入 。 因 此，Round(1.5)=2; Round(2.5)=2， 
Round(0.15)=0.2，Round(0.005)=0。 一 一 译 者 注 
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口 在 PHP 中 调用 函数 时 ， 函 数 名 和 封装 参数 的 圆 括 号 之 间 可 以 有 (也 可 以 没有 ) 空格 ， 两 种 
写法 都 正确 : 
round ($num); 
round ($num); 

口 number_format () 函数 接受 两 个 可 选 参数 ， 这 两 个 参数 分 别 用 来 指定 使 用 什么 字符 代 
表 小 数 点 位 数 和 千 分 号 。 这 是 非常 有 用 的 ， 例 如 ， 在 某 些 文化 的 习惯 中 ， 将 1 000.89 写 
为 1.000 89。 如 果 和 希望 使 用 这 些 选 项 ， 请 参看 PHP 手 册 中 相关 的 语法 描述 。 

4.4 理解 优先 级 


在 讨论 不 同类 型 的 算术 运算 符 时 ,必定 会 涉及 优先 级 的 问题 。 优 先 级 是 指 一 系列 计算 的 执行 
顺序 。 例 如 ， 下 面 变 量 的 值 是 多 少 ? 
snumber = 10 - 4/ 2; 


$number 的 值 是 3(10 减 去 4 等 于 6, 然后 被 2 除 ) 还 是 8 (4 被 2 除 后 等 于 2, 10 再 减 去 2 等 于 8) ? 


这 里 的 


结果 是 8， 因 为 除法 的 优先 级 要 高 于 减法 。 


附录 B 中 提供 了 PHP 里 运算 符 优先 级 的 完整 列表 (包括 这 里 尚未 介绍 到 的 运算 符 )。 然 而 ， 可 


以 使 用 


加 括号 对 来 绕 过 所 有 的 概念 和 规则 , 而 不 是 去 尝试 记 住 那个 有 特殊 字符 的 庞大 表格 。 圆 括 





号 对 的 优先 级 通常 要 高 于 所 有 其 他 运算 符 。 因 此 : 


snumber = (10 - 4) / 2; // 3 
Snumber = 10 - (4 / 2); // 8 


在 算式 中 使 用 圆 括号 能 够 确保 不 会 看 到 因为 优先 级 问题 引发 的 特殊 (错误 ) 结果 ,也 可 以 使 
用 圆 括号 来 将 复杂 的 算式 重 写 为 更 少 行 数 的 代码 。 让 我 们 使 用 圆 括号 重 写 handle_calc.php 脚 
本 ， 将 多 行 代码 合并 为 一 行 ， 同 时 确保 精度 。 

[> 管理 优先 级 

















(1) 在 文本 编辑 器 或 者 IDE 中 打开 handqle_calc.php (参看 脚本 4-3 ) 。 
(2) 改变 总 费用 第 一 次 计算 的 方式 (参看 脚本 4-4): 


Stotal = ((Sprice * Squantity) + S$shipping) - S$discount; 








脚本 4-4 使 用 圆 括号 能 够 改写 压缩 多 行 算式 (参看 脚本 4-3)， 而 不 会 影响 脚本 的 算术 精度 


PoPODODP 


Oo 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 


<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Product Cost Calculator</title> 
<style type="text/css" media="screen"> 
.number { font-weight: bold;} 
</style> 
</head> 
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11 <body> 

12 <?php // 脚本 4-4 - handle_calc.php #3 

13 /* 脚本 从 calculator.html 获 取 数 值 

14 并 计算 总 成 本 和 月 付款 数 。*/ 

六 区 

16 // 可 以 在 此 进行 错误 处 理 。 

17 

18 // 从 $_POST 数 组 获取 数值 : 

19 Sprice = $_POST['price']; 

20 Squantity = $_POST['gquantity']; 

21 S$discount = $_POST['discount']; 

22 Stax = $_POST['tax']; 

23 S$shipping = $_POST['shipping']; 

24 S$payments = $_POST['payments']; 

25 

26 // 计算 总 额 : 

27 Stotal = (($price * S$quantity) + $shipping) - $discount; 

28 

29 // 定义 税率 : 

30 S$taxrate = ($tax/100) + 1; 

3 

32 // 纳入 税率 后 的 总 成 本 : 

33 Stotal = $total * staxrate; 

34 

35 // 计算 月 付款 数 : 

36 Ss$monthly = Stotal / Spayments; 

37 

38 // 格式 化 数值 ， 打 印 2 位 小 数 。 

39 Stotal = number_ format ($total, 2); 

40 smonthly = number_ format ($monthly, 2); 

二 二 

42 // 打印 结果 : 

43 print "<p>You have selected to purchase:<br/> 

44 <span class=\"number\">$quantity</span> widget(s) at <br/> 

45 $<span class=\"number\">$price</span> price each Plus a <br/> 

46 $<span class=\"number\">$shipping</span> Shipping cost and a <br/> 

47 <span class=\"number\">$tax</span> percent tax rate.<br/> 

48 After your $<span class=\"number\"> $discount</span> discount, the total cost is 

49 $<span class=\"number\">$total </span>.<br/> 

50 Divided over <span class=\"number\"> S$payments</span> monthly payments, that 
would be $<span class=\"number\">s$monthly </span> each.</p>"; 

与 下 

52 2 

53 <ZBody> 


54 </html> 

只 要 使 用 圆 括 号 确保 算术 运算 能 够 正确 执行 , 就 不 用 逃避 将 所 有 的 运算 放 在 一 个 步 又 中 。 另 
外 一 个 选择 是 记 住 PHP 中 多 个 运算 符 的 优先 级 规则 ， 但 是 使 用 圆 括号 还 是 要 简便 一 些 。 

(3) 改变 税 值 的 计算 方式 : 

$Staxrate = ($tax/100) + 1; 


这 里 将 原 有 的 2 行 税 值 计算 代码 合并 为 1 行 。 
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(4) 在 同 calculator .htm1l 相 同 的 目录 下 保存 脚本 ， 并 且 在 Web 浏 览 器 上 测试 〈 参 见 图 4-8 
和 图 4-9) 。 


Fill out this form to calculate the total cost: 








Price: 1.50 
Quanttty |250 


Discount: 0 





You have selected to purchase: 
Taxl6 |(%) 250 widgetfs) at 
$1.S0 price each plus a 


Shippi thod: | | need ityesterday! Y 
PPME MO Pe ee ey $19.365 shipping cost and a 









































Number of payments to make: 2 6& percent tax rate. 
Btter your $0 discount, the total costis $418.02. 
Divided over 2 monthly payments, that would be $209.01 each. 
图 4-8 ”再次 测试 表单 图 4-9 ”尽管 运算 被 精简 , 但 是 计算 结果 却 没 有 发 生 改 
变 。 如果 看 到 不 同 的 结果 或 者 得 到 一 个 错误 消 




















息 , 就 要 再 次 检查 圆 括号 的 匹配 情况 (打开 和 
关闭 的 圆 括 号 数量 应 当 一 致 ) 











Vv 提示 

口 在 创建 算式 时 ， 要 确保 圆 括号 对 始终 都 是 匹配 的 〈 每 个 打开 的 圆 括号 都 需要 有 一 个 关闭 

的 圆 括号 ) 。 否 则 将 会 出 现 解析 错误 。 

口 倘若 使 用 在 这 里 应 用 的 这 个 方法 ， 可 以 将 所 有 的 总 费用 计算 合并 为 只 有 1 行 代码 (而 不 是 
3 行 )， 但 是 这 样 会 让 一 些 事情 过 于 简单 化 。 


4.5 数值 的 自 增 和 自 减 


如 同 Per 和 其 他 大 多 数 编程 语言 一 样 ，PHP 包 含 很 多 快捷 方式 ， 用 来 避免 产生 一 些 不 美观 的 
结构 ， 例 如 : 


Stax = Stax + 1; 
当 某 个 变量 的 值 需要 增加 1 (被 成 为 增 量 调整 ) 或 者 减少 1 (被 成 为 减 量 调整 ) 时 ， 可 以 使 用 
++ 或 者 --， 下 面 分 别 给 出 二 者 的 示例 : 


Svar = 20; // 20 
Svar++; /7/ 21 
Svar++; // 22 
$var--; // 21 


这 里 再 一 次 重 写 handle_calc .php 肢 本， 专门 测 试 这 个 概念 。 

之 变量 值 的 自 增 

(1) 在 文本 编辑 器 或 者 IDE 中 打开 handle_calc.php (参看 脚本 4-4)。 
(2) 改变 脚本 4-3 中 税率 的 计算 方法 ， 更 改 结果 如 下 所 示 (参看 脚本 4-5): 
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Staxrate = $tax/100; 
Staxratet++; 


脚本 4-5 数值 的 自 增 和 自 减 是 一 种 常见 的 运算 方式 ， 它 们 分 别 用 ++ 或 者 -- 表 示 




















工 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
用 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
3 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Product Cost Calculator</title> 

学 <style type="text/css" media="screen"> 

8 .number { font-weight: bold;} 

9 </style> 

10 </head> 

11 <body> 

12 <?php // 脚本 4-3 - handle calc.php #4 

13 /* 该 脚本 从 calculator.html 获 取 数 值 

14 并 计算 总 成 本 和 月 付款 数 。*/ 

15 

16 // 可 以 在 此 进行 错误 处 理 。 

17 

18 // 从 $_POST 数 组 获取 数值 : 

19 S$price = $_POST['price']; 

20 Squantity = $_POST['gquantity']; 

21 S$discount = $_POST['discount']; 

22 Stax = $_POST['tax']; 

23 S$shipping = $_POST['shipping']; 

24 Spayments = $_POST['payments']; 

25 

26 // 计算 总 额 : 

27 Stotal = ((Sprice * S$Squantity) + $shipping) - S$discount; 

28 

29 // 定义 税率 : 

30 S$taxrate = $tax/100; 

31 S$taxrate++; 

32 

33 // 纳入 税率 后 的 总 成 本 : 

34 S$total = Stotal * S$taxrate; 

35 

36 // 计算 月 付款 数 : 

37 Ss$monthly = Stotal / Spayments; 

38 

39 // 格式 化 数值 ， 保留 2 位 小 数 。 

40 Stotal = number_ format ($total, 2); 

41 Smonthly = number_ format ($monthly, 2); 

42 

43 // 打印 结果 : 

44 print "<p>You have selected to purchase:<br/> 

45 <span class=\"number\">$quantity</span> widget(s) at <br/> 

46 $<span class=\"number\">$price</span> price each Plus a <br/> 
47 $<span class=\"number\">$shipping</span> shipping cost and a <br/> 
48 <span class=\"number\">$tax</span> percent tax rate.<br/> 

49 After your $<span class=\"number\"> $discount</span> discount, the total cost is 
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50 S$<span class=\"number\">S$Stotal </span>.<br/> 
51 Divided over <span class=\"number\"> S$payments</span> monthly payments, that 
would be $<span class=\"number\">$monthly </span> each.</p>"; 


53 ?> 
54 </body> 
55 </html> 


第 一 行 通过 将 $tax 的 值 除 以 100 计 算出 了 税率 。 第 二 行将 该 结果 加 1， 使 其 与 总 值 相 乘 后 能 
够 得 到 带 税 的 总 值 。 

(3) 在 同 calculator .html 相 同 的 目录 下 保存 脚本 ,并 在 Web 浏 览 嚣 中 进行 测试 (参见 图 4-10 
和 图 4-11)。 














Fl out this form to calculate the total cost: 
Price: 5.00 
Quantity: 100 


Discount: 10.00 





You hawve selected to purchase: 
100 wideetrs) at 

$FS.00 price each Plus a 

$5.00 shipping cost and a 


Tax: |7.5 (0) 


Shipping method: | Slowand steady 六 




















Number of paytments to make: 10 TS5 percent ta rate. 
Lfter your $10.00 discount, the total cost 1s $5S32.13. 
Divided over 10 monthly payments, that would be $53.21 each. 
图 4-10 ”最 后 一 次 执行 表单 图 4-11 使 用 长 的 或 者 短 的 变量 自 增 版 本 ， 将 不 会 对 








计算 结果 产生 影响 (比较 脚本 4-4 和 脚本 4-5) 
ww 提示 
口 虽然 从 功能 上 看 ， 编 写 代码 既 可 以 使 用 $staxrate=$taxrate+1; 方 法 ， 也 可 以 使 用 精简 
的 Staxrate++， 但 后 一 种 方式 (使 用 自 增 操作 符 ) 更 加 专业 并 且 通 用 。 
口 在 第 6 章 中 ， 将 看 到 自 增 操作 符 如 何 被 普遍 地 同 循环 联合 使 用 。 


算术 赋值 运算 符 
nae oggnogogoogogooggoriouri gga000000 
有 加 加 0 
ee 
DODOUDhandle calc.php O0000000000 


Sos SPOS tl/ 
So O00 sO00S 
Stax += 1; // 1.05 


加 




















4.6 00000 75 





4.6 创建 随机 类 
本 章 介 绍 的 最 后 一 个 国 数 是 *and() ， 它 是 一 个 随机 数 生成 工具 ， 用 于 输出 随机 数 : 


Si = rand (ss.//-31 
sn = rand(); // 87 


如 果 需 要 将 产生 的 数值 限定 在 一 个 特定 范围 内 ，rand () 国 数 可 以 通过 接受 最 小 值 和 最 大 值 
的 参数 做 到 


sn = rand (0, 10); 


参数 的 值 在 包含 范围 内 ， 因 此 在 上 面 的 例子 中 0 和 10 都 可 能 是 返回 的 值 。 
让 我 们 创建 一 个 简单 的 脚本 “Lucky Numbers” 作 为 随机 数 生成 的 示例 。 | 


之 创建 随机 数 的 步 又 


(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 文档 ， 命 名 为 random.php (参看 脚本 4-6): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Lucky Numbers</title> 

</head> 

<body> 



































脚本 4-6 zand() 函数 生成 随机 数 


1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 

7 

8 

9 








<title>Lucky Numbers</title> 
</head> 
<body> 
<?php // 脚本 4-6 - random.php 
/* 这 个 脚本 生成 3 个 随机 数 。*/ 


// 可 以 在 此 进行 错误 处 理 。 


14 // 创建 3 个 随机 数 : 
15 S$nl = rand (1, 99); 


16 $n2 = rand (1, 99); 
17 $n3 = rand (1, 99); 
18 





19 // 打印 随机 数 : 

20 print "<p>Your lucky numbers are:<br/> 
21 Snlibr/> 

22 $n2<br/> 

23 Sn3</p>"™; 


76 0040 0000 





25,Y 澡 > 
26 </body> 
27 </html> 


(2) 如 果 需 要 ， 如 下 添加 PHP 标 签 和 错误 管理 : 
<?php // 脚本 4-6 - random.php 
(3) 创建 3 个 随机 数 : 


snl 
sn2 
$n3 


这 些 数字 通过 3 次 单独 调用 rand () 函数 产生 ， 并 且 将 每 个 

(4) 打印 这 些 数字 : 

print "<p>Your lucky numbers are:<br/> 

$sn1l<br/> 

$n2<br/> 

$n3</p>"; 

这 个 print 语 名 非常 简单 。 数 字 被 打印 出 来 , 通过 在 它们 之 前 加 上 一 个 HIML 的 break 标 签 每 
一 个 数字 都 在 不 同 的 行 里 显示 。 

(5) 关闭 PHP 代 码 和 HTML 页 面 : 


</body> 
</html> 


(6) 将 脚本 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 保存 为 random.php， 然 后 在 Web 浏 览 器 中 
进行 测试 (参见 图 4-12 和 图 4-13)。 








rand: ‘1, 99)3 
rand. ‘(1; 99)» 
rand (1, 99); 























全 


果 赋 值 给 了 不 同 的 变量 。 
































Your lucky rnmbers are: Four lucky numbers are: 
32 23 
B68 81 
71 2 
图 4-12 通过 调用 rand() 函数 图 4-13 再 次 运行 脚本 , 将 获得 
产生 了 3 个 随机 数 不 同 的 结果 


w 提示 

D getrandmax() 函数 将 返回 使 用 rana () 可 能 产生 的 随机 数 中 最 大 的 数值 。 结 果 的 值 会 因 

为 所 使 用 的 操作 系统 的 不 同 而 有 所 不 同 。 

口 PHP 有 另外 一 个 用 以 生成 随机 数 的 国 数 mt_rand() 。 它 同 *anda() 非常 相似 (但 是 更 好 )， 
并 且 在 对 于 生成 密码 这 样 的 敏感 场景 来 说 是 更 好 的 选择 。 请 查看 PHP 手 册 以 查找 关于 
mt_rand() 国 数 的 更 多 和 更 完整 的 主题 讨论 。 
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其 他 的 算术 函数 

PHPUDOOOOOO0O000000000000000UD zcound()D numper 
fozrmat()[ rand()[ 

PHPU round O00000000000ceiid}00000000000000000 
oo 0 O000000t0000090000000000000000" 0“ 
DODO 

加 可 可 本 器 
go0go 


snumber alssy( 22 /3 
snumber SE (DDI) 7 


和 
UUOUO000UPHPUUOUOUOUOUOOUOOOOOOOUOOUOUU0OO0O0O0O00D0 
UDUPHFEUOUUOOUOOOOOD 





4.7 回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
com/forum/。 














4.7.1 回顾 


D 四 个 主要 的 算术 运算 符 是 什么 ? 
D 下 面 的 代码 为 什么 不 能 工作 : 
print "The total is $$total"; 
如 何 修改 ? 
口 如 果 HTML 页 面包 括 要 提交 到 PHP 肢 本 的 表单 ， 为 什么 必须 通过 URL 加 载 该 HTML? 
D 哪些 函数 可 以 用 来 格式 化 数值 ? 如何 对 数值 截取 特定 位 数 的 小 数 。 
D 运算 符 优先 级 的 重要 性 是 什么 ? 
口 自 增 、 自 减 运算 符 是 什么 ? 
口 有 哪些 算术 赋值 运算 符 ? 














4.7.2 ”实践 


口 在 PHP 手 册 中 查找 一 个 本 音 新 学 的 函数 。 单 击 页 面 上 的 链接 ,浏览 其 他 与 数值 有 关 的 函数 。 
口 创建 一 个 新 的 包含 数值 的 HTML 表 单 。 然 后 创建 PHP 脚 本 接收 表单 数据 ， 执 行 一 些 计 算 ， 
格式 化 数值 ， 并 打印 结果 。 














本 章 内 容 

口 创建 HTML 表单 

口 连接 字符 串 

口 处 理 换行 符 

口 HTML 和 PHP 

口 字符 串 的 编码 和 解码 
口 查找 子 字符 串 

口 替换 局 部 字符 串 

口 回顾 和 实践 


第 2 章 已 介绍 过 ，PHP 中 使 用 的 第 二 种 变量 类 型 是 字符 囊 ， 它 是 用 单 引 号 或 者 双 引 号 括 起 来 
的 一 系列 的 字符 。 一 个 字符 串 变 量 可 以 包含 一 个 单独 的 字母 、 一 个 单词 、 一 个 句子 、 一 个 段落 、 
HTML 代码 ， 甚 至 是 一 些 没 有 意义 的 字符 、 数 字 和 符号 〈 可 能 代表 一 串 密码 )。 字 符 串 也 许 是 PHP 
中 最 常用 的 变量 类 型 。 

本 章 将 涵盖 PHP 中 最 基础 的 内 置 函 数 和 操作 字符 串 数据 的 运算 符 ， 不 论 这 些 字符 串 是 来 自 
于 表单 还 是 由 脚本 最 初 声明 的 。 本 章 还 将 介绍 一 些 常见 技术 。 例 如 ， 字符 串 修 剪 〈 如 去 掉 空 白 
和 其 他 预定 义 字符 )、 字 符 串 连接 和 字符 串 编码 。 字 符 串 的 其 他 用 法 将 在 后 续 的 章节 中 诠释 。 


5.1 创建 HTML 表单 


如 同 第 3 章 , 首先 创建 一 个 HTML 表 单 , 这 个 表单 将 向 PHP 脚 本 传送 一 些 字符 串 变 量 的 值 。 这 
里 用 以 诠释 理论 的 示例 是 一 个 在 线 的 公告 板 或 者 论坛 ， 用 户 可 以 在 上 面 发 布 信息 、 他 们 的 Email 
地 址 和 姓名 (参见 图 5-1)。 


之 创建 一 个 HTML 表 单 




















(1) 在 文本 编辑 器 或 者 IDE 中 创建 一 个 新 的 HTML 文 档 ， 命 名 为 posting.html (参看 脚本 
5-1): 
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<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3 .org/TRV/xhtm11/DTD/xhtm1l1-ttransitional .Qtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Forum Posting</title> 

</head> 

<body> 

<1-- 脚本 5$-1 - posting.html --> 

<div><p>Please complete this form to submit your posting:</p> 





了 Please complete this form to subtmit your posting: 
First ame 
Last Name: 


Email Address: 





Posting: 


5-1 这 个 HTML 表 单 是 本 章 大 多 数 示例 的 基础 














| 














脚本 5-1 这 个 表单 向 PHP 脚 本 传送 字符 串 数据 














工 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

三 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Forum Posting</title> 

7 </head> 

8 <body> 

9 <!-- 脚本 5-1 - posting.html --> 

10 <div><p>Please complete this form to submit your posting:</p> 

11 

12 <form action="handle post.php" method="post"> 

13 

14 <p>First Name: <input type="text" name="first name" size="20" /></p> 
1 

16 <p>Last Name: <input type="text" name="last name" size="20" /></p> 
Bi 

18 <p>Email Address: <input type="text" name="email" size="30" /></p> 
19 

20 <p>Posting: <textarea name="posting" rows="9" cols="30"></textarea></p> 
21 

22 <input type="submit" name="submit" value="Send My Posting" /> 
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24 
25 
26 
27 


</form> 
</div> 

</body> 
</html> 


(2) 创建 表单 的 初始 标签 : 

<form action="handle post.php" method="post"> 

这 个 表单 将 会 将 数据 用 POST 的 方式 传送 给 handle_post .php 脚 本 。 
(3) 为 First Name、Last Name 和 Email Address 添 加 文本 框 标 签 : 








<p>First Name: <input type="text" name="first name" size="20" /></p> 
<p>Last Name: <input type="text" name="last name" size="20" /></p> 
<p>Email Address: <input type="text" name="email" size="30" /></p> 


这 些 都 是 第 3 章 中 介绍 过 的 基本 文本 输入 类 型 。 请 记 住 ， 不 同文 本 框 标签 的 name 值 应 当 


符合 PHP 变 量 的 命名 规则 (不 能 有 空格 、 
意 组 合 )。 
(4) 为 posting 添 加 一 个 文本 框 标签 : 
<p>Posting: 
posting 字 上段 是 一 个 textarea (文本 


(5) 创建 提交 按钮 并 关闭 表单 : 


<input type="submit" name="submit" 





<textarea name="posting" 


不 能 以 数字 开头 、 只 能 是 字母 、 数 字 和 下 划 线 的 任 


rows="9" cols="30"></textarea></p> 


区 )， 它 提供 比 文本 输入 框 更 大 的 输入 空间 。 





value="Send My Posting" /> 


见 符 


</form> 
每 个 表单 都 必须 有 一 个 提交 按钮 (或 提交 
(6) 完成 HTML 页 面 : 


</div> 
</body> 
</html> 


0) 将 文件 保存 为 posting .html， 放置 在 启用 了 PHP 的 服务 器 上 的 适当 目录 中 ,并 在 Web 浏 





图 片 )。 











中 进行 查看 (参见 图 5-1)。 
这 是 一 个 HTML 页 面 ， 因 此 它 不 必 在 启用 了 PHP 的 服务 器 上 查看 。 但 是 由 于 它 最 终 会 向 PHP 


脚本 发 送 数 据 ， 因 此 最 好 还 是 将 它 保 存在 服务 器 上 。 


Vv 提示 

口 从 技术 上 说 ， 除 了 上 传 的 文件 之 外 ， 所 有 表单 数据 都 会 被 作为 字符 串 发 送 给 处 理 脚 本 。 
这 包括 在 文本 框 中 输入 的 数值 数据 、 被 选中 的 下 拉 菜 单 选项 、 复 选 框 或 者 单 选 按钮 的 值 ， 
等 等 。 例 如 ， 第 4 章 的 表单 ， 将 带 有 数值 的 字符 串 发 送 到 脚本 。 

口 很 多 用 PHP 编 写 的 论坛 系统 都 是 可 以 免费 使 用 的 。 本 书 将 不 涉及 如 何 进 行 完整 的 论坛 开 

发 ， 但 是 可 以 在 我 编写 的 《PHP 6 与 MySQL 5 基础 教程 》 中 找到 多 语言 论坛 的 开发 方法 。 

口 本 书 的 Web 站 点 中 有 一 个 供 读者 发 布 问题 和 其 他 读者 (以 及 作者 ) 回答 问题 的 论坛 ， 可 以 
在 www.LarryUllman.com/forum/list.php?30 找 到 。 
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5.2 连接 字符 串 


连接 (concatenation) 是 一 个 难以 驾驭 ， 但 却 是 非常 有 用 的 概念 。 它 用 来 将 不 同 的 项 串联 到 
一 起 。 在 编程 时 ， 特 指 字符 囊 (string) 的 连接 。 它 的 运算 符 是 句点 (. )， 使 用 方法 如 下 : 


SS 和 AH TO 
8S2 = WOCLIQTT7 
Sgreeting = SSs1 . $s2; 


连接 的 最 后 结果 是 Sgreeting 变 量 的 值 为 Hello，world!。 

由 于 PHP 处 理 变 量 的 特有 方式 ， 因 此 下 面 的 方法 可 以 达到 同样 的 效果 : 

$greeting = "Ssi$s2"; 

这 是 因为 在 PHP 处 理 变 量 时 ， 将 放置 在 双 引 号 中 的 变量 用 它们 的 值 进行 了 替代 。 但 是 ， 推 
荐 使 用 句点 连接 字符 串 的 正式 方法 ， 并 且 这 种 方法 使 用 得 更 加 广泛 (这 将 更 明显 地 看 出 代码 中 




















发 生 了 什么 )。 
另外 一 种 能 够 起 到 连接 作用 的 方法 是 使 用 连接 赋值 运算 符 : 
Sgreeting = 'Hello, ‘' 
Sgreeting .= 'world!'; 


第 二 行 大 致 的 意思 是 “为 Sgreeting 赋 予 它 当 前 的 值 ， 并 且 同 worla1! 连 接 ”。 最 后 的 结果 是 
$sgreeting 的 值 为 Hello,，world!。 

posting.html 脚 本 向 handle_post .php 页 面 发 送 了 几 个 字符 串 变 量 。 在 这 些 变 量 中 ,， 交 
辑 上 应 该 将 姓 和 名 连接 起 来 。 推 荐 使 用 像 在 表单 中 那样 提供 用 户 姓名 分 开 填 写 的 方式 , 这 种 方式 
应 用 得 更 加 普遍 。 另 外 ， 这 种 方式 更 加 有 利于 将 两 个 部 分 看 作 一 个 人 名 。 在 编写 PHP 脚 本 的 时 个 
请 注意 对 此 加 以 考虑 。 

信使 用 连接 


























(1) 在 文本 编辑 器 或 者 IDE 中 新 建 一 个 文档 ， 命 名 为 handle_post .php (参看 脚本 5-2) : 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Forum Posting</title> 

</head> 

<body> 





最 常见 的 字符 串 变 量 操 作 ， 可 以 将 它 看 作 是 字符 串 的 加 法 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Forum Posting</title> 


脚本 5-2 该 PHP 脚 本 展示 了 连接 
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</head> 
<body> 
<?php // 脚本 5-2 - handle_post.php 
/* 脚本 从 posting .html 接 收 5 个 值 : 
first name, last name, email, 


// 可 以 在 此 进行 错误 处 理 。 


// 从 $_POST 数 组 获取 数值 : 

$sfirst name = $_POST['first name']; 
$last name = $_POST['last name']; 
$sposting = $_POST['posting']; 


IMANrRLDNPO 





Dr 
Oo 


// 创建 姓名 变量 : 


$name $first name . ' "' 


. $last name; 


DD 
DP 


// 打印 一 条 信息 : 
print "<div>Thank you, 
<p>$posting</p></div>"; 


$sname, 


?> 
</body> 
</html> 


NNDNDNN N 
OO 
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(2) 创建 初始 PHP 标 签 ， 如 果 有 必要 ， 开 局 错误 管理 : 


<?php // 脚本 5-2 - handle_post.php 

如 果 没 有 启用 aisplavy_errors， 或 者 erzor_ 
在 这 里 添加 适当 的 代码 以 改变 这 些 设置 。 

(3) 将 表单 中 的 数据 赋值 给 局 部 变量 : 


$sfirst_name $_POST['first _ name']; 
$last_name $_POST['last_ name']; 
$sposting = $_POST['posting']; 


表单 使 用 POST 方式 进行 数据 传输 ， 因 








posting, submit */ 


for your posting: 








reporting 的 级 别 设置 错误 ， 请 参见 第 3 章 ， 


此 所 有 的 表单 数据 在 $_PosT 中 都 将 可 用 。 


这 个 示例 没有 包含 Email 相关 的 代码 ， 这 是 因为 用 不 着 它 ， 但 是 也 可 以 复制 这 些 代 码 来 引用 





这 个 值 。 
(4) 使 用 连接 创建 一 个 新 的 Sname 变 量 ; 
Sname = $first name . ' ' . $last_ name; 





在 这 里 , 连接 操作 将 两 个 变量 和 一 个 空格 合并 在 一 起 , 从 而 创建 了 一 个 名 为 Sname 的 新 变量 。 
假设 输入 Elliott 和 Smith 作 为 两 个 人 名 输入 框 的 值 ， 那 么 $name 的 值 将 是 E11liott Smith。 


(5) 向 用 户 输 出 消息 : 


print "<div>Thank you, 
<p>$posting</p></div>"; 


这 条 消息 向 用 户 报 告 了 在 表单 中 输入 的 内 容 。 
(6) 关闭 PHP 代 码 段 并 完成 HTML 页 面 : 


Sname， 


for your posting: 
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?> 
</body> 
</html> 


(7) 将 脚本 保存 为 handle_post .php， 并 放置 在 同 posting .ntml 相 同 的 目录 下 (在 启用 了 
PHP 的 服务 器 上 )， 并 且 在 Web 浏 览 器 上 测试 表单 和 脚本 (参见 图 5-2 和 图 5-3)。 

















Please complete this form to submit your posting: 
First Name: Jeremy 

Last Name: Messersmith 

Email Address: jm(@example.org 


This is my posting. It could be 
more original. 









































Posting: Thank you, Jeremy hlesserstmith, for your postineg: 
This 1s rmy posting. It could be more original. 
图 5-2 使 用 中 的 HTML 表 单 图 5-3 ”PHP 页 面 运行 结果 

















请 注意 ， 必 须 通 过 URL 来 加 载 表 单 (http://something ) ， 这 样 当 提 交 表单 时 ， 用 来 处 理 的 PHP 
脚本 也 会 通过 URL 来 运行 。 
V 提示 
D 如 果 在 表单 中 使 用 不 同类 型 的 引号 并 且 看 到 打印 结果 中 有 多 余 的 斜 线 ， 请 参看 第 3 章 相 
“Magic Quotes” 中 对 于 这 种 问题 的 解释 和 处 理 方法 。 
口 请 注意 , 理解 PHP 中 单 引 号 和 双 引 号 的 不 同 是 非常 重要 的 。 在 单 引 号 中 的 字符 将 被 按照 字 
面 处 理 ， 而 在 双 引 号 中 的 字符 将 会 被 解释 (interpreted) 〈 例 如 ， 一 个 变量 的 名 称 将 会 被 它 
的 值 所 代 禁 )。 请 复习 第 3 章 中 的 相关 内 容 。 
口 可 以 按照 需要 将 许多 字符 串 连 接 起 来 。 其 至 还 可 以 将 数字 连接 成 为 字符 串 : 
Snew_string = $s1 . $s2 . $number; 
这 是 因为 PHP 属 于 弱 类 型 (weakly typed) 编程 语言 ， 这 就 意味 着 它 的 变量 并 不 会 被 锁定 
为 一 种 特定 的 格式 。 在 这 里 ，$number 变量 将 会 被 转换 成 为 字符 捉 并 且 附 加 在 
$new_string 变 量 的 值 中 。 
口 连接 有 很 多 种 使 用 方式 ， 甚 至 当 为 函数 提供 参数 的 时 候 。 例 如 下 面 这 个 不 常见 但 却 实 
用 的 示例 : 
$text = nl2br($heading . $body); 
n12br () 函数 在 第 1 章 简 要 介绍 过 ，5.3 布 将 会 详细 介绍 这 个 函数 。 
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5.3 ”处 理 换 行 符 


字符 串 中 的 换行 符 通常 给 PHP 开 发 新 手 带 来 一 些 问 题 。 用 户 可 以 在 textarea 表 单元 素 中 用 





敲 击 Return 或 者 Enter 的 方式 输入 多 行文 本 。 每 次 襄 击 Retum 或 者 Enter 的 结果 在 字符 串 中 都 相当 于 
产生 一 个 换行 符 。 这 些 换行 符 在 textarea 中 会 起 作用 ， 但 是 在 PHP 页 面 呈 现 中 将 不 会 产生 任何 


效 


四 
不 








(参见 图 5-4 和 图 5-5)。 








了 Please complete this form to submit your posting: 
First Name: Rocky 

Last Name: |Yotolato 

Email Address: rv@example.edu 


Here's one line. 
Here's another line. 


Here's a third line. 
































posting Thank you, Rocky Votolato, for your posting: 
Here's one line. Here's another line. Here's a third line. 
图 5-4 表单 数据 中 文本 区 里 的 换行 符 图 5-5 换行 符 没有 被 Web 浏 览 器 呈现 出 来 




















为 了 能 够 在 Web 页 面 呈现 时 与 换行 符 有 等 同 的 效果 , 可 以 使 用 break 标 签 : <br/>。 季 和 运 的 是 ， 


PHP 中 的 n12br () 函数 能 够 将 换行 符 自动 转换 成 为 break 标 签 : 


Svar = nl2br ($var); 


我 们 将 在 handle_post .php 中 应 用 这 个 国 数 ， 以 便 用 户 提交 的 内 容 能 够 保持 其 原 有 格式 。 
这 转换 换行 符 








(1) 如 果 handle_post .php (参看 脚本 5-2) 不 在 开启 状态 的 话 ， 在 文本 编辑 器 上 打开 它 。 
(2) 当 为 Sposting 变 量 赋 值 时 应 用 n12br () 函数 (参看 脚本 5-3): 


$sposting = nl2br($_POST['posting']); 





脚本 5-3 通过 使 用 n12br () 函数 ，Web 浏 览 器 正确 显示 了 发 布 文 本 区 中 的 换行 符 


小 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

:: <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

S <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Forum Posting</title> 

A </head> 

8 <body> 

9 <?php // 脚本 5-3 - handle_post.php #2 

10 /* 脚本 从 posting .html 接 收 5 个 值 : 

11 first name, last name, email, posting, submit */ 
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// 可 以 在 此 进行 错误 处 理 。 


$sfirst name = $_POST['first name']; 


2 
3 
14 
15 // 从 $_POST 数 组 获取 数值 : 
6 
7 Slast name = $_POST['last name']; 
8 





$sposting = nl2br($ POST['posting']); 
19 
20 // 创建 姓名 变量 : 
21 Sname = $first name . ' ' . $last_name; 
22 


23 // 打印 一 条 信息 : 
24 print "<div>Thank you，Sname，for your posting: 
25 <p>$posting</p></div>"; 


27 ?> 
28 </body> 
29 </html> 








现在 $posting 将 被 赋值 为 $_POST ['posting'], Thank you, Rocky Wotolate, for your posting: 
其 中 所 有 的 换行 符 都 被 转换 成 为 HTML 的 break 标 签 。 

(3) 保存 文件 ， 放 置 在 同 bosting .html 相 同 的 目录 
(在 启用 了 PHP 的 服务 器 上 ) 中 ， 并 且 在 Web 浏 览 器 中 进 | Heres another ne 


Here's one line. 






































行 测试 (参见 图 5-6)。 Here's a third line. 
w 提示 图 5-6” 同 图 $-4 中 相同 的 提交 数据 
口 还 能 够 通过 在 双 引号 之 间 放 置换 行 符 (\n) 的 方 ee 
式 向 字符 串 中 插入 换行 符 。 
D 其 他 的 HTML 标 签 (如 <p> 标 签 ) 同样 能 够 影响 Web 呈 现 页 面 的 布局 。 可 以 使 用 一 个 替 


换 国 数 将 换行 符 (或 其 他 字符 ) 转换 为 <p> 标 签 , 但 是 这 样 做 的 代码 比 直接 调用 n12bz () 

















要 复杂 得 多 。 
口 字符 串 中 的 换行 符 发 送 到 浏览 器 后 会 产生 效果 , 但 效果 只 会 出 现在 HTML 的 源 代码 中 ( 参 
见 图 5-7)。 





<qdiv>Thank you, Rocky Votolato, for your posting: 
<p>Here's one line. 


Here's another line. 


Here's a third line.</p></div></hody> 
</html> 


图 5-7 ”图 5-5 页 面 的 HTML 源 代码 显示 了 Web 浏 览 器 中 包含 换行 符 的 
效果 ( 即 ， 在 HTML 源 代码 中 加 入 空白 ) 





























5.4 HTML 和 PHP 


正如 本 书 在 此 之 前 多 次 提 及 的 那样 ，PHP 是 一 项 被 频繁 用 来 向 Web 浏 览 器 发 送 数据 的 服务 器 
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端 技术 。 这 些 数据 可 以 是 纯 文本 、HTML 代 码 或 者 两 者 都 有 的 形式 。 

在 本 章 的 主要 示例 里 ， 数 据 将 从 一 个 HTML 表 单 中 输入 ， 然 后 使 用 PHP 在 Web 浏 览 器 上 打印 
出 来 。 一 个 潜在 的 问题 是 , 用 户 可 以 在 表单 中 输入 HTML 字 符 , 这 将 会 对 页 面 的 格式 产生 影响 ( 参 
见 图 5-8 和 图 5-9)， 或 许 还 会 导致 更 糟 的 结果 ， 从 而 引发 安全 方面 的 问题 。 








Please complete this form to subtnit your posting: 
First Name: IDamien 
Last Name: Rice 


Email Address: dr(@example.net 





























es ele ens eres Dieb: Thank you, Damien Rice, for your posting: 
<li>Something</ 1i> . 
<li>Something Else</1i> Let's make an ordered list: 
<1i>Something New</1i> 
</ul> 
e Something 
Posting: e Something Else 
» Something New 
图 5-8 ”如 果 用 户 在 发 布 时 输入 图 $-9 ” 当 再 次 打印 时 它 被 Web 
HTML 代 码 浏览 器 呈现 出 来 








可 以 使 用 一 些 PHP 函 数 来 处 理 PHP 字 符 串 变量 中 的 HTML 标 签 。 
DD htmlspecialchars () 将 特定 的 HIML 标 签 转换 为 实体 版 本 。 
口 ntmlentities() 将 所 有 的 HTML 标 签 转 换 为 实体 版 本 。 
D strip_tags () 移 除 所 有 的 HTML 和 PHP 标 签 。 

前 两 个 函数 将 HTML 标 签 (如 <span>) 转换 为 实体 版 本 ,如 &lt; span&gt ;。 这 个 实体 版 本 
在 输出 时 出 现 但 是 不 被 呈现 。 如果 希望 显示 代码 而 不 发 布 , 可 以 使 用 它们 其 中 的 任何 一 个 来 做 到 。 
第 3 个 函数 strip_tags () 用 来 完全 移 除 所 有 的 HTML 和 PHP 标 签 。 

出 于 两 个 原因 这 里 应 当 查 看 在 用 户 提 供 的 数据 中 是 否 有 特殊 的 标签 ,首先 ,正如 已 经 提 到 的 ， 
用 户 提 交 的 HTML 可 能 会 对 页 面 的 呈现 产生 影响 (参见 图 5-10) (例如 , 表格 错位 、 扭 曲 CSS 样 式 ， 
或 者 只 是 添加 不 应 该 出 现 的 格式 )。 第 二 个 原因 更 加 重要 。 由 于 JavaScript 是 被 放置 在 HTML 标 签 
中 的 ， 因 此 怀 有 恶意 的 用 户 会 提交 JavaScript， 当 它们 重新 显示 在 页 面 上 时 ， 这 些 脚 本 会 被 执行 。 
这 就 是 跨 站 点 脚本 (cross-site scripting，XSS) 攻击 的 实现 方式 。 




















| 四 


全 You just executed my ]ava5cript code! 








图 $-10 ”在 Web 浏 览 器 中 显示 用 户 提 交 的 HTML 会 产生 很 恶劣 的 后 果 ， 如 执行 JavaScript 脚 本 
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为 了 了 解 这 些 函 数 的 影响 ， 下 面 在 对 handle_post .php 的 重 写 过 程 中 将 逐个 使 用 它们 并 展 
现 备 自 的 结果 。 


-> 处 理 HTML 和 PHP 


(1) 如 果 handle_post .php 不 是 开启 状态 的 话 ， 请 在 文本 编辑 器 或 IDE 中 将 它 打 开 (参看 肢 
本 5-3)。 
(2) 在 print 行 之 前 添加 下 面 的 代码 行 (参看 脚本 5-4): 


$shtml_post = htmlentities($_POST ['posting']); 
$strip_post = strip tags($_POST ['posting']); 








脚本 5-4 ”这 个 版 本 的 PHP 脚 本 用 两 种 不 同 的 方式 处 理 HTML 标 签 














1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
3 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Forum Posting</title> 

7 </head> 

8 <body> 

9 <?php // 脚本 5-4 - handle_post.php #3 

10 /* 脚本 从 posting .html 接 收 5 个 值 : 

11 first name, last name, email, posting, submit */ 

12 

13 // 可 以 在 此 进行 错误 处 理 。 

14 

15 // 从 $_POST 数 组 获取 数值 : 

16 S$first name = $_POST['first name']; 

17 S$last name = $_POST['last name']; 

18 Sposting = nl2br($_ POST['posting']); 

19 

20 // 创建 姓名 变量 : 

21 Sname = S$first name . ' ' . $last_ name; 

22 

23 // 处 理 变量 中 的 HITML 标 签 : 

24 $html post = htmlentities($ POST ['posting']); 

25 $strip post = strip tags($ POST ['posting']); 

26 

27 // 打印 一 条 信息 : 

28 print "<div>Thank you, $name, for your posting: 

29 <p>Original: $posting</p> 

30 <p>Entity: $html post</p> 

31 <p>Stripped: $strip post</p></div>"; 

32 

33 ?> 

34 </body> 


35 </html> 
为 了 搞 清楚 这 两 个 函数 之 间 的 区 别 , 将 它们 都 应 用 于 posting, 并 在 过 程 中 创建 两 个 新 的 变 
量 。 本 书 在 这 里 使 用 $_PosT['posting'] 而 不 是 Sposting， 是 因为 Sposting 已 经 反映 了 
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n12br () 函数 的 应 用 ， 这 意味 着 有 可 能 引用 用 户 没 有 显 式 输入 的 break 标 签 。 
(3) 将 print 语 句 按照 下 面 的 方法 改写 : 


print "<div>Thank you, S$name, for your posting: 
<p>Original: S$posting</p> 

<p>Entity: $html_post</p> 

<p>Stripped: S$strip post</p></div>"; 


为 了 突出 结果 的 不 同 ， 打 印 出 3 个 不 同 的 posting 版 本 。 第 一 个 是 与 输入 时 相同 的 原始 
posting, 第 二 个 是 对 posting 应 用 了 htmlentities() 函数 后 的 版 本 。 它 将 显示 HTML 标 签 
而 不 呈现 它们 的 作用 。 最 后 一 种 是 使 用 了 strip_tags () 的 版 本 , 结果 中 没有 任何 HTML (或 
PHP) 标签 。 

(4) 保存 文件 ， 将 其 放 在 同 posting .html 相 同 的 目录 下 〈 在 启用 了 PHP 的 服务 器 上 )， 并 且 
在 Web 浏 览 器 上 再 次 进行 测试 (参见 图 5-11 和 图 5-12)。 


了 Please complete this form to submit your posting: 


























First Name: Laura 
Last Name: Burhenn 
Email Address: lb@example.com 


I don't understand why it says 
<em>something</ em>. 





Thank you, Laura Burhenn, for your posting: 


Ongmal. I don't understand why 1 says something. 


























Posting Enhty: 工 dent understand why it says <em>somethng</em>. 
Stnipped: I don't understand why tt says something. 
图 5-11 ”作为 posting 的 一 部 分 所 输入 的 图 5-12 ”PHP 结果 页 显示 了 没有 被 修改 过 的 原始 的 
HTML 字 符 将 被 PHP 处 理 posting， 以 及 应 用 了 htmlentities() 








和 strip_tags() 后 的 效果 
如 果 查 看 结果 PHP 页 面 的 HTML 源 代码 (参见 图 5-13)， 还 将 看 到 应 用 这 些 函 数 后 的 效果 。 


<p>Original: I don't understand Why it says <em>something</em> .</p> 
<p>Entity: I don't understand why it says g&lt;emgot;somethingelt;/emgeot; .</p> 
<p>Stripped: I don't understand why it says something.</p></div></hody> 


图 5-13 ”图 为 图 5-10 中 所 示 内 容 的 HTML 源 代码 






































Vv 提示 

口 出 于 安全 的 考虑 ， 用 htmlentities()、htmlspecialchars() 或 者 strip_tags() 来 
处 理 需要 在 Web 浏 览 器 上 打印 的 用 户 提 供 的 任何 数据 , 通常 情况 下 这 是 很 好 的 做 法 。 因 为 
本 书 的 目的 是 最 小 化 可 能 带 来 的 复杂 度 ， 因 此 这 里 没有 这 样 做 。 

口 html_entity _dqecodqe () 国 数 同 htmlentities() 国 数 正好 相反 ， 它 将 HIML 实 体 转换 

为 相应 的 HTML 代 码 。 

口 另外 一 个 向 Web 浏 览 器 输出 字符 串 的 函数 是 wordwrap () 。 这 个 函数 按照 指定 长 度 折 行 处 
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理 字符 串 。 


口 为 了 将 换行 符 转 换 为 break 标 签 并 且 同 时 移 除 所 有 HTML 或 者 PHP 标 签 ， 可 以 在 strip_ 


tags () 图 数 后 使 用 nl12br () : 


Sposting = nl2br(strip_ tags($_POST['posting'])); 


在 这 行 代码 中 ，strip_tags () 国 数 将 被 首先 调用 ， 并 且 它 的 结果 会 被 发 送 给 n12pr () 


国 数 。 


5.5 字符 串 的 编码 和 解码 





第 3 章 的 3.6 市 中 展示 了 如 何 使 用 GET 方 法 将 数据 附加 在 URL 中 向 页 面 传送 。 在 示例 中 ,没有 
使 用 真正 的 表单 ， 而 是 将 数据 直接 加 在 URL 上 ,通过 URL 发 送 到 接收 的 脚本 。 准 确 地 说 ,这 种 方 
式 只 能 传送 一 个 单词 ,， 且 不 能 带 有 空格 和 标点 。 但 是 如 果 希 望 传 送 多 个 词 的 变量 值 或 特殊 字符 时 


怎么 办 呢 ? 
urlencode() 国 数 可 以 通过 URL 安 全 地 将 任意 值 传送 到 PHP 


郑 本 。 顾 名 思 义 ， 这 个 国 数 接受 





一 个 字符 串 ， 并 对 之 编码 (encode) (改变 它 的 格式 ) ， 以 便 它 完全 适合 作为 URL 的 一 部 分 传输 。 


这 个 函数 用 加 号 (+) 替换 掉 空 格 ， 并 且 将 特殊 字符 (如 省 略 号 ) 
可 以 编写 以 下 代码 来 使 用 这 个 函数 : 


$string = urlencode ($string); 


转换 为 较 少 出 现 问 题 的 形式 。 


为 了 展示 urlencode () 函数 的 应 用 ， 我 们 重 写 handle_post .php 页 ， 添 加 一 个 链接 ， 用 来 


向 第 3 个 页 面 传送 用 户 的 姓名 和 Bmail 地 址 。 








(1) 如 果 handle_post .php 不 在 开启 状态 的 话 ， 请 在 文本 编辑 器 或 者 IDE 中 打开 它 (参看 脚 











="enn lang= "en"> 


本 5-4) o 

(2) 删除 在 之 前 那些 步骤 中 添加 的 htmlentities() 和 strip_tags() 代码 行 (参看 脚本 
5-5) 。 
脚本 5-5 该 脚本 在 将 两 个 变量 添加 进 链接 之 前 先 对 它们 进行 编码 。 通 过 使 用 这 种 方法 ， 变 量 被 

成 功 地 发 送 给 其 他 页 面 

1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

各 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 

6 <title>Forum Posting</title> 

7 </head> 

8 <body> 

9 <?php // 脚本 5-5 - handle_post.php #4 

10 /* 脚本 从 posting .html 接 收 5 个 值 : 

11 first name, last name, email, posting, submit */ 
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// 可 以 在 此 进行 错误 处 理 。 


$first name = $_POST['first name']; 
$last _ name = $_POST['last name']; 


2 
3 
14 
15 // 从 $_POST 数 组 获取 数值 : 
6 
8 S$Sposting = nl2br($_POST['posting']); 





20 // 创建 姓名 变量 : 
21 Sname = S$first name . ' ' . $last_ name; 
22 


23 // 打印 一 条 信息 : 
24 print "<div>Thank you, S$name, for your posting: 
25 <p>$posting</p></div>"; 


27 // 创建 连接 另 一 页 面 的 链接 : 

28 $name = urlencode ($name); 

29 S$Semail = urlencode($ POST['email']); 

30 print "<p>Click <a href=\"thanks .php? 
name=$name&temail=$email \">here</a> to continue.</p>"; 


31 
32. ?> 
33 </body> 


34 </html> 
(3) 恢复 到 调用 print 代 码 的 较 老 版 本 。 


print "<div>Thank you, S$name, for your posting: 
<p>$posting</p></div>"; 


(4) 在 print 语 句 后 ,添加 下 面 的 代码 : 


Sname = urlencode (Sname) ; 
Semail = urlencode($_ POST['email']); 


该 脚本 将 会 向 第 2 个 页 面 传送 这 两 个 变量 。 因 此 ， 这 两 个 变量 必须 经 过 编码 。 
因为 之 前 还 没有 引用 或 者 使 用 过 $email 变 量 ， 第 二 行 代码 不 仅 从 $_POST 数 组 中 检索 emai1l 
的 值 ， 而 且 还 在 这 一 步 中 对 它 进 行 了 编码 。 这 同 下 面 两 行 分 开 的 代码 执行 效果 是 一 样 的 : 


Semail = $_POST['email']; 
$email = urlencode ($email); 


(5) 添加 另外 一 个 创建 链接 的 print 语 句 : 
print "<p>Click <a href=\"thanks.php ?name=$name&email=$email\"> 
”here </a> to continue.</p>"; 


print 语 句 的 核心 意图 是 在 这 个 Web 页 面 中 创建 一 个 HTMLIL 链 接 , 其 源 代码 将 和 下 面 的 代码 
类 似 : 

<a href="thanks.php?name=Larry+ Ullman&email=larry%40example.com">here</a> 

为 了 实现 这 个 目标 ， 需 要 硬 编码 大 部 分 HIML， 并 且 包 含 适 当 的 变量 名 称 。 由 于 HTML 代码 
需要 将 链接 的 URL 放 置 在 双 引 号 中 ,并 且 print 语 句 已 经 使 用 了 双 引 号 ， 因 此 必须 对 它们 进行 转 
义 以 便 将 它们 打印 出 来 。 
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图 





(6) 保存 文件 ,将 其 放置 在 启用 了 PHP 的 服务 器 的 适当 目录 下 ,并 且 在 Web 浏 览 器 上 测试 ( 参 
5-14 和 图 5-15)。 





了 Please complete this form to submit your posting: 
First Name: Christopher 

Last Name: O'Reilly 

Email Address: |chris.oreilly@example.com 


Nothing like a piano cowver of 
Radiohead or Elliott Smith! 





Thank you, Christopher O'Reilly, for your posting: 


















































Posting Nothing like a piano cover of Radiohead or Elhott Stmithl 
Click here to continue. 
图 5-14 表单 的 另 一 次 应 用 图 5-15 现在 处 理 脚 本 显示 了 到 另外 
一 个 页 面 的 链接 


也 请 注意 ， 点 击 这 个 链接 将 导致 一 个 服务 器 错误 ， 因 为 还 没有 编写 thanks .php 脚 本 。 
0) 查看 处 理 页 面 的 HTML 源 代码 以 查看 在 HTML 代 码 中 的 结果 链接 (参见 图 5-16)。 
































<p>Click <a href="thanks.php?name=Christopher+0%27Reillygemail=chris.oreillys40example.com">here</a> 




















图 15-16 ”图 5-15 的 源 代码 ， 显 示 动 态 生 成 的 链接 





w 提示 

口 从 表单 直接 发 送 的 值 ， 在 被 发 送 之 前 会 自动 进行 URI 一 编码 ， 接 收 脚本 收 到 后 再 对 其 进行 

解码 。 因 此 ， 你 只 需要 使 用 urlencode() 函数 手动 编码 数据 即 可 〈 如 本 节 例 子 所 示 )。 

口 urldecode () 函数 同 urlencode() 正 好 相反 ， 它 接受 一 个 编码 过 的 URL 并 且 将 之 转换 
回 标准 形式 。 它 的 使 用 并 不 会 那么 频繁 ， 这 是 因为 PHP 能 够 自动 将 它 获取 到 的 大 部 分 值 
进行 解码 。 

口 因为 可 以 对 函数 使 用 连接 ， 所 以 新 的 print 语 句 可 以 按照 下 面 的 方法 进行 编写 : 


print 'Click <a href="thanks.php? name=' . $name . 
—'&email=' . S$email . '">here</a> to continue.'; 


这 种 方法 有 两 个 额外 的 好 处 : 首先 它 使 用 单 引 号 标识 语 名 的 开始 和 结束 ， 这 就 意味 着 不 
用 做 双 引 号 的 转 义 操作 ， 基 次， 使 用 的 变量 更 加 显而易见 ， 它 们 不 会 淹没 在 一 大 堆 其 他 
的 代码 中 。 
口 不 用 为 了 在 URL 中 使 用 数值 的 PHP 值 而 对 其 进行 编码 , 因为 它们 并 不 包含 会 出 现 问 题 的 字 
符 。 尽 管 如 此 ， 对 它们 进行 编码 也 不 会 造成 什么 影响 。 



































Email 地 址 。 
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Thank ¥ou, Chnstopher 心 卫 elly. 
We wil contact you at chtris.creily 多 exatmble.cotm. 


图 $-17 ”这 是 第 3 个 页 面 ， 会 在 本 章 结尾 时 创建 ， 该 页 面 打印 从 URL 中 接收 到 的 信息 
字符 串 的 加 密 和 解密 
DDOO0OOOQ000000000000 加 窜 D encrypttDUUOUOOUOODODOUUOOODOOD 
ggg 
D00000EmA000000U0UUUUUUO 
Nass ll ll ee 
器 0 wa 
0 
加 gg Ogg 
sdata = erypt(Sdata) 
DOUUUUUUmrypt encrypt UUUUUUDmrypt decrypt ()0 OUUO 
D000000000000000000UU0Mcrypt UUUUPHPIUUUUOUUUO 
D0000000000000000 PHPSAdvanced: Visual QuickPro Guide[Peachpit Press[] 
200700000000000UD 
DOOO00000000000000000UU0U0UUUUU My7SQLU PostgreSQDD 
Oracle] D SQL Server O0000000000000000U0U000000UU0 
上 加 


5.6 查找 子 字符 串 


PHP 有 一 些 函 数 可 以 用 来 拆 解 、 搜 索 以 及 比较 字符 串 。 尽 管 这 些 函数 通常 都 要 在 一 定 条 件 下 
使 用 ， 我 们 将 在 第 6 章 中 讨论 这 些 条件 ， 但 是 这 些 函 数 是 非常 重要 的 ， 本 书 在 这 里 一 定 要 对 它们 
进行 介绍 ， 第 6 章 会 更 加 正式 地 运用 它们 。 

在 本 章 的 较 前 部 分 已 经 学 习 到 如 何 连接 字符 串 。 除了 用 短小 的 字符 串 片段 组 成 更 大 的 字符 串 
外 ,还 可 以 从 字符 串 中 提取 出 某 一 部 分 。 使 用 方法 提取 部 分 字符 串 的 技巧 是 ， 必 须 对 字符 串 本 身 
有 所 了 解 ， 以 便 能 够 有 效 地 执行 这 样 的 操作 。 

strtok() 函数 使 用 一 个 预定 的 分 隔 符 作为 标记 〈 例 如 ， 喜 号 或 者 空格 ) ， 从 大 字符 串 中 创 
建 子 字符 串 。 例 如 ， 如 果 用 户 在 一 个 字段 中 输入 他 们 的 全 名 (假定 他 们 的 姓 和 名 使 用 空格 进行 分 
隔 )， 可 以 通过 这 样 的 代码 确定 他 们 的 名 : 

$sfirst = strtok($_ POST['name'], ' '); 

这 行 代码 告诉 PHP 从 $_PosT[I'name'] 中 获取 直到 遇 到 空格 之 前 的 所 有 内 容 。 

如 果 你 得 到 的 用 户 以 “Surname，First” 这 种 格式 输入 的 全 名 ， 你 需要 通过 下 面 的 代码 找到 
他 的 姓 : 
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$1Last = strtok($_ POST['name'], ', '); 

第 三 种 提取 部 分 字符 串 的 方法 是 引用 字符 串 中 字符 的 索引 位 置 (indexed position)。 字符 串 的 
索引 (index) 是 一 个 字符 的 位 置 值 ， 从 字符 串 开 头 数 起 。 但 是 ， 如 同 大 多 数 编程 语言 一 样 ，PHP 
所 有 的 索引 都 是 以 0 开始 的 。 例 如 ， 字 符 串 Larry 的 索引 中 ， 工 在 位 置 0 上 ， 接 下 来 的 a 的 索引 为 1， 
I 为 2， 第 二 个 r 为 3，y 为 4。 虽 然 字符 串 Larry 的 长 度 为 5 个 字符 ， 但 是 它 的 索引 是 从 0 到 4 的 〈 也 就 
是 索引 通常 是 从 0 到 字符 串 长 度 减 1)。 

了 解 这 些 后 ， 可 以 根据 子 字符 串 中 字符 所 在 的 索引 位 置 ， 利 用 substz () 函数 来 创建 子 字符 
串 ， 例 如 ; 

$sub = substr($string, 0, 10); 

第 一 个 参数 是 需要 提取 子 字符 串 的 主 字符 串 。 第 二 个 参数 指明 子 字符 串 从 哪里 开始 提取 , 也 
就 是 它 在 主 字 符 串 中 的 索引 位 置 (0 的 意思 是 希望 从 第 一 个 字符 开始 ) 。 第 三 个 参数 的 意思 是 从 起 
始点 开始 ， 需 要 为 子 字符 串 提 取 的 字符 数量 (10)。 如 果 主 字符 串 不 够 10 个 字符 的 长 度 ， 子 字符 
串 将 以 主 字 符 串 的 结尾 作为 结束 。 这 个 参数 是 可 选 的 ， 如 果 省 略 ， 子 字符 串 也 将 以 主 字符 串 的 结 
尾 作为 结束 。 

还 可 以 使 用 负数 来 从 字符 串 末尾 处 开始 计数 


$sstring = 'ardvark'; 
$sub = substr($string, -3, 3); // ark 


第 二 行 的 意思 是 , 返回 的 结果 是 从 字符 串 的 末尾 处 倒数 第 3 个 字符 开始 的 3 个 字符 。 就 本 例 来 
说 ， 省 略 第 3 个 参数 也 可 以 得 到 相同 的 结果 : 

$sub = substr($string, -3); // ark 

可 以 使 用 strlen () 来 了 解 字 符 串 中 字符 的 数量 : 

print strien('Hello, world!'); // 13 

该 函数 在 计算 数量 的 时 候 将 包括 空格 和 标点 符号 ,而 str_word_count () 函数 是 用 来 获取 字 
符 串 中 单词 数量 的 。 它 同 substr () 一 起 ， 将 用 在 handle_post .php 肢 本 的 下 一 个 版 本 中 。 












































字符 串 比较 
二 
O00g000 
日 stxems《 加 
日 stznatcmp() 加 加 回回 回回 加 加 加 加 加 四 加 回国 
DODOOOODODOOODOODODODDO strcasecmp () 吕 stzrnatcasecmp ()[ 
台中 ogo 
DOO0go 
避 zxzsexz 人 Ogggggogong 人 ggggggogngggoogon 人 ggggn 
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DQ stzpos(OUUUUO0OU0O0O0U0O0O0O000O0000000 
UUOUUOU0O0O0O0O0U0O0U0O0O0000000 stristz()DU stzipos(O0U0000 
加 加 加 


这 创建 子 字 符 串 








(1) 如 果 handle_post .php 是 未 开启 状态 ， 在 文本 编辑 器 或 IDE 中 打开 它 (参看 脚本 5-5)。 
(2) 在 print 语 句 之 前 ， 添 加 下 面 的 代码 (参看 脚本 5-6 ) : 


Swords = str word count($posting); 





脚本 5-6 该 版 本 的 handle_post .php 返 回 发 布 内容 中 的 单词 数量 ， 并 且 限 制 显 示 的 内 容 仅 为 


\D oawm 心 wmN 


oawm 必 wh 





DDNDNIN 
(HB. AD 


DL 
心 


DDND N N N 
\D ou 


LO WO mw mw UW 
ODPoOo 


LO CU 
ON Un 


开头 的 50 个 字符 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0o0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Forum Posting</title> 

</head> 

<body> 

<?php // 脚本 5-6 - handle post.php #5 

/* 脚本 从 posting .html 接 收 5 个 值 : 

first name, last name, email, posting, submit */ 


// 可 以 在 此 进行 错误 处 理 。 


// 从 $_POST 数 组 获取 数值 。 

$sfirst name = $_POST['first name']; 
$last name = $_POST['last name']; 
Sposting = nl2br($_POST['posting']); 


// 创建 姓名 变量 : 


Sname = S$first name . ' ' . S$last name; 


// 计算 单词 数量 : 
$words = str word count($posting); 


// 将 发 送 的 内 容 截 断 一 部 分 : 
$posting = substr($posting, 0, 50); 


// 打印 一 条 信息 : 

print "<div>Thank you, $name, for your posting: 
<p>$posting...</p> 

<p> ($words words)</p></div>"; 


?> 
</body> 
</html> 
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在 这 个 版 本 的 脚本 中 ,希望 对 用 户 发 布 的 内 容 做 两 件 事情 ,一 个 是 显示 内 容 包含 的 单词 数量 。 


这 个 信息 在 这 里 收集 ， 然 后 赋值 给 $words 变 量 。 


例 ， 








(3) 在 下 一 行 代码 中 (同样 也 在 print 语 句 之 前 ) 添加 如 下 代码 : 

S$posting = substr($posting, 0, 50); 

这 里 希望 脚本 做 的 第 二 件 事 是 限制 所 显示 的 内 容 为 开头 的 50 个 字符 。 可 以 这 样 应 用 这 个 示 
例如 ， 如 果 一 个 页 面 显示 发 布 内 容 的 开始 部 分 ， 然后 提供 给 用 户 通 往 完整 内 容 的 链接 。 为 了 

















执行 显示 限制 ， 这 里 调用 substr () 函数 。 


oo 


(4) 依照 下 面 的 代码 更 新 print 语 句 : 


print "<div>Thank you, S$name, for your posting: 
<p>s$posting...</p> 
<p> ($words words)</p></div>"; 


这 里 有 两 处 变化 。 首 先 在 posting 后 添加 了 省 略 号 ,这 表明 这 只 是 整个 posting 内 容 的 一 部 
然后 在 另外 一 段 中 打印 单词 的 数量 。 

(5) 删除 urlencode () 和 相应 的 print 两 行 代码 。 

这 里 指 的 是 在 之 前 的 脚本 中 添加 的 链接 thanks .php 脚 本 的 代码 。 

(6) 保存 文件 , 将 其 放 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 , 并 且 再 次 在 Web 浏 览 器 中 测试 


(参见 图 5-18 和 图 5-19)。 








了 Please complete this form to submit your posting: 
First Name: Regina 

Last Name: Spektor 

Emall Address: rs@example.edu 


This is a longer post. This is 
a longer post. This is a longer 
post. This is a longer post. 

This is a longer post. This is 
a longer post. This is a longer 
post. This is a longer post. 

This is a longer post. This is 















































a longer post. This is a longer Thank you, Feaina Spektor, for your postine: 
post. This is a longer post. 
Posting: This is a longer post. This 1s a longer post. This 
(oa 
图 5-18 发布 的 内 容 多 于 50 个 字符 图 5-19 ”显示 的 内 容 将 被 剪 短 ,单词 的 
数量 也 会 被 显示 出 来 
V 提示 


口 如 果 想 检验 一 个 字符 串 是 否 匹 配 某 些 特定 的 格式 (如 是 否 是 有 效 的 Email 地 址 ), 需要 使 用 
正则 表达 式 。 正 则 表达 式 (regular expression) 是 一 种 高 级 概念 ， 可 以 用 它 来 定义 模式 并 
且 检 验 某 个 值 是 否 符合 这 种 模式 。 参 见 PHP 手 册 或 我 编写 的 《PHP 6 与 MySQL 5 基础 教 
程 》 了 解 更 多 信息 。 
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5.7 ”和 蔡 换 局 部 字符 串 


我 们 不 仅 需要 在 一 个 字符 串 中 查找 子 字 符 串 〈 如 本 章 之 前 部 分 所 讨论 的 那样 ) ， 还 可 能 需要 
用 新 的 值 蔡 换 子 字符 串 (replace substring)。 可 以 使 用 str_ireplace() 国 数 来 实现 : 

SSstring = Str_irepbplace(Sneedle，Sreplacement， Shaystack) ; 

每 次 Sneedle 在 $haystack 出 现时 ， 这 个 函数 都 将 之 替换 为 Sreplacement。 例 如 : 

$me 

$me 

现在 变量 sme 的 值 变 为 Larry Edward Ullman。 

该 函数 执行 的 搜索 不 区 分 大 小 写 。 为 了 增加 限制 性 ， 可 以 使 用 str_replace() 进 行 区 分 大 小 
写 的 搜索 。 在 下 一 个 示例 中 ， 将 使 用 str_ireplace() 来 在 提交 的 文本 中 除去 “脏话 ”。 

这 里 介绍 与 字符 串 有 关 的 最 后 一 个 函数 trim()。 这 个 函数 用 来 移 除 在 字符 串 首尾 处 的 所 
有 空白 : 空格 、 换 行 符 和 制 表 符 。 在 一 个 字符 串 变量 中 有 多 余 的 空格 是 非常 常见 的 ， 这 或 许 
是 由 于 用 户 输入 信息 时 的 不 小 心 , 或 者 由 于 不 正规 的 HTML 代 码 造成 。 出 于 简洁 、 数 据 完整 性 
和 Web 设 计 的 考虑 ,需要 在 使 用 这 些 字符 串 之 前 删除 空白 。 向 Web 浏 览 器 发 送 多 余 的 空白 会 使 
页 面 看 起 来 古怪 ， 如 果 是 发 送 给 数据 库 或 者 Cookie， 终 将 在 日 后 造成 令 人 遗憾 的 结果 (例如 ， 
如 果 密 码 中 含有 多 余 的 空白 ， 那 么 当 输 入 不 带 有 这 些 空白 的 密码 时 将 不 会 被 正确 匹配 )。 








'Larry E. Ullman'; 
str_ireplace('E.', 'Edward', S$me); 


























调整 字符 串 的 大 小 写 

加 me 
国人 
四 大 可 es 
D strtoupper OOUUUUUUUUUO 
D strtolower DUUUUUUUUUUUO 

国 
DOO0O000000000000000000000000000000000UU0UUO 
Oggogg00U0o0 


trim() 函数 自动 从 字符 串 的 开头 和 结尾 处 (而 不 是 中 间 ) 去 除 所 有 的 空白 。trim() 的 使 用 
格式 如 下 : 


$string = ' extra space before and after text ' 
sstring = trim($string); 
// $string 现在 等 于 'extra space before and after text'. 


信使 用 str_ireplace() 


(1) 如 果 handle_post .php 是 未 开启 状态 ， 在 文本 编辑 器 或 IDE 中 打开 它 (参看 脚本 5-6)。 
(2) 对 表单 数据 应 用 trim() 函数 (参看 脚本 5-7): 
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$sfirst name = trim($_ POST['first name']); 
$last_ name = trim($_POST['last_ name']); 
$posting = trim($_POST['posting']); 


脚本 5-7 这 是 对 脚本 进行 处 理 的 最 后 一 个 版 本 ， 应 用 了 trim() 国 数 ， 并 且 用 一 些 X 来 代替 脏话 


工 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

污 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 
5 

6 

对 

8 








<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Forum Posting</title> 
</head> 
<body> 
<?php // 脚本 5-7 - handle_post.php #6 
10 /* 脚本 从 posting.html 接 收 5 个 值 : 
11 first name, last name, email, posting, submit */ 


13 // 可 以 在 此 进行 错误 处 理 。 





15 // 从 $_POST 数 组 获取 数值 。 

16 // 使 用 zim() 去 掉 多 余 的 空格 : 

17 S$first name = trim($ POST['first name']); 
18 S$last name = trim($_ POST['last name']); 
19 S$posting = trim($ POST['posting']); 


20 

21 // 创建 姓名 变量 : 

22 Sname = S$first name . ' ' . $last_ name; 
23 


24 // 计算 单词 数量 : 

25 S$words = str word count (S$posting); 

26 

27 // 去 掉 脏 话 : 

28 S$posting = str ireplace('badword', 'XXXXX', S$posting); 
29 

30  // 打印 一 条 信息 : 

31 print "<div>Thank you, S$name, for your posting: 

32 <p>$posting</p> 

33 <p>($words words)</p></div>"; 


34 
35. ?> 
36 </body> 


37 </html> 


只 有 当 收 到 的 数据 的 首尾 包含 多 余 的 空白 ，trim() 才 被 应 用 。 
(3) 删除 使 用 substr () 的 代码 段 。 


$sposting = substr($posting, 0, 50); 


这 里 希望 在 这 个 示例 中 看 到 完整 的 发 布 内 容 ， 因 此 删除 了 对 substr () 的 调用 。 
(4) 在 print 语 句 之 前 ,添加 下 面 的 代码 : 


Sposting = str_ireplace('badword', 'XXXXX', Sposting); 
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这 个 特殊 的 示例 找 出 了 在 发 布 内 容 中 的 脏话 并 将 它们 去 除 。 这 里 的 代码 用 padword 来 代 赫 实 
际 中 的 脏话 (当然 可 以 使 用 任何 你 想 使 用 的 词语 )。 
如 果 希 望 禁 换 更 多 的 脏话 ， 可 以 使 用 下 面 的 多 行 代码 来 实现 : 


$sposting = str_ireplace('badwordl1', 'XXXXX', Sposting); 
$sposting = str_ireplace('badword2', 'XXXXX', Sposting); 
Sposting = str_ireplace('badword3', 'XXXXX', S$posting); 


(5) 更 新 Print 语 句 以 便 它 不 再 使 用 省 略 号 : 


print "<div>Thank you, S$name, for your posting: 
<p>$posting</p> 
<p> ($words words)</p></div>"; 


(6) 保存 文件 , 将 其 放 在 启用 了 PHP 的 服务 器 的 适当 目录 下 , 并 在 Web 浏 览 器 上 再 次 进行 测试 
(参见 图 5-20 和 图 5-21)。 














了 Please complete this form to subrmit your posting: 
First Name: Bad 

Last Name: |Poster 

Email Address: faker@bad.example.com 


I feel like using a BADWORD in 
my post! 





Thank you, Bad Poster, for your posting: 






































Posting: Tfeel lkee using a SIOOOEC in my postl 
Send My Posting (9 words) 
图 5-20 ”如 果 用 户 输入 了 不 希望 他 们 图 5-21 ”那么 可 以 让 PHP 来 替换 
使 用 的 词 它们 
V 提示 


口 str_ireplace() 国 数 将 捕捉 到 内 容 中 的 脏话 。 例 如 ， 如 果 输 入 Ifeel like using badwords， 

结果 将 会 是 I feel like using XXXXXs。 

口 stz_ireplace() 国 数 还 能 够 接受 一 个 子 字符 串 的 数组 、 一 个 替换 字符 串 的 数组 ， 甚 至 
一 个 主 字符 串 的 数组 。 因 为 你 也 许 此 时 并 不 知道 数组 是 什么 ， 所 以 在 这 里 将 不 对 这 个 技 
术 进 行 解释 。 

口 如 果 需 要 去 除 一 个 字符 串 开 头 或 结尾 的 空白 ， 而 不 是 两 者 都 需要 去 除 ，PHP 将 trim() 国 

数 分 为 两 个 更 加 特殊 的 函数 : rtrim() 移 除 字 符 串 变量 结尾 处 的 空白 ， 而 1trim() 则 处 

理 字符 串 开 头 的 空白 。 两 者 一 起 用 的 效果 等 同 于 trim()。 


sstring 
sstring 



































rtrim($string); 
ltrim($string); 
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5.8 回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
com/forum/。 














5.8.1 回顾 


口 如 何 创建 一 个 字符 串 ? 
口 单 引 号 与 双 引 号 在 使 用 上 有 什么 不 同 ? 
口 什么 是 连接 操作 符 ? 什么 是 连接 赋值 操作 符 ? 
口 将 包括 换行 符 的 字符 串 输出 到 浏览 器 会 有 什么 影响 ?如 何 将 换行 符 转换 成 pr 标签 。 
口 如 果 表 单元 素 的 值 会 在 浏览 器 中 显示 , 而 在 该 表单 元 素 中 输入 的 是 HIML 可 能 导致 什么 加 
题 ? 此 时 有 什么 办 法 处 理 提 交 的 表单 数据 ? 5 
口 哪个 函数 可 以 使 数据 安全 地 包含 在 URL 中 传递 ? 
口 如 何 去 除 字符 串 中 的 问题 字符 ?如 果 不 去 除 问 题字 符 会 出 现 什么 问题 ? 
口 字符 串 中 字符 索引 的 起 始 值 是 什么 ? 
口 trim() 函数 的 功能 是 什么 ? 


5.8.2 ”实践 


口 在 PHP 和 手册 中 查找 一 个 本 章 介 绍 的 新 函数 ， 利 用 页 面 中 的 链接 查看 其 他 字符 串 相 关 函 数 。 
口 在 PHP 和 手册 中 查找 substr () 函数 ， 并 查看 该 函数 的 其 他 示例 ， 加 深 理解 substr () 的 使 
用 方法 。 

口 编写 脚本 5-5 中 要 用 到 的 thanks .php 脚 本 。 如 果 需 要 帮助 ， 可 以 参考 第 3 章 中 的 
hello .php 脚 本 (参见 脚本 3-7)。 

口 重 写 handle_post .php (参见 脚本 5-7) 最 终 版 中 的 print 语 句 , 使 用 单 引 号 和 连接 代 赫 
双 引 号 。 

口 创建 一 个 新 的 HTML 表 单 ， 接 收 字符 串 值 。 接 下 来 ， 创 建 一 个 PHP 脚 本 ， 接 收 表 单 中 的 数 
据 ， 处 理 任 意 HTML 或 PHP 代 码 ， 以 某 些 方式 操作 数据 ， 并 打印 出 结果 。 




















控制 结构 








本 章 内 容 

口 创建 HTML 表单 
口 i£ 条 件 语句 

口 验证 函数 

口 使 用 else 

口 更 多 运算 符 

口 使 用 elseif 

口 switch 条 件 语句 
口 fo 循环 

0 回顾 和 实践 





控制 结构 是 指 条 件 和 循环 ， 它 是 编程 语言 的 主要 组 成 部 分 。PHP 包 含 两 种 常见 的 条 件 控制 语 
句 : if 和 switch， 本 章 将 介绍 如 何 使 用 它们 。 你 可 以 使 用 条 件 控 制 语句 来 测试 某 个 变量 或 表达 
式 是 否 满足 某 一 特定 的 条 件 , 并 基于 测试 的 结果 执行 相应 的 操作 。 这 些 功能 能 够 用 来 构建 更 具 动 
态 性 的 Web 网 站 。 

讨论 i1£ 条 件 的 讨论 将 引入 最 后 的 两 类 运算 符 : 比较 运算 符 和 逻辑 运算 符 (在 之 前 的 章 市 中 已 
经 介绍 了 算数 运算 符 和 赋值 运算 符 ) 。 通 常 这 些 运算 符 还 将 连同 布尔 概念 中 的 真 / 假 概念 一 起 运用 。 

最 后 ,本 章 将 开始 在 编程 中 使 用 循环 ， 它 可 以 以 一 定 的 友 代 次 数 重 复 同 一 操作 。 循 环 能 够 节 
省 很 多 编程 的 时 间 ， 在 第 7 章 中 ， 还 将 看 到 它 同 数组 联合 使 用 获得 更 为 丰富 的 功能 。 


6.1 创建 HTML 表单 


如 同 在 之 前 的 章节 中 那样 ， 本 章 的 示例 将 基于 一 个 向 PHP 页 面 发 送 数据 的 HIML 表 单 。 在 这 
个 例子 中 ， 表 单 是 一 个 简单 的 注册 页 面 ， 它 需要 以 下 信息 (参见 图 6-1): 
口 Email 地 址 ; 
口 密码 ; 
口 密码 确认 ， 
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口 生日 〈 用 于 验证 年 龄 )， 
口 喜欢 的 颜色 (用 于 定制 外 观 )，; 
口 网 站 条 款 协 议 (常见 需求 )。 








Please complete this form to register: 





Email Address: 

Password: 

Confirm Password: 

Year You Were Born: yyyy | 
Favorite Color: [ Pick One 剖 


口 Iagree to the terms (whatever they may be). 








E Register ) 
We 








图 6-1 本章 中 使 用 的 HTML 表 单 
接 下 来 的 步骤 将 在 涉及 PHP 代 码 之 前 先 处 理 表 单 。 
过 创建 HTML 表 单 


(1) 在 文本 编辑 器 或 者 IDE 中 新 建 一 个 HTML 文 档 ， 命 名 为 register.html (参看 脚本 6-1): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0o0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Registration Form</title> 

</head> 

<body> 

<1-- 脚本 6-1 - register.html --> 

<div><p>Please complete this form to register:</p> 




















脚本 6-1 这 个 虚构 的 注册 表单 是 本 章 的 示例 基础 











和 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

奸 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Registration Form</title> 

gy </head> 

8 <body> 

9 <!-- 脚本 6-1 - register.html --> 

10 <div><p>Please complete this form to register:</p> 

和 

12 <form action="handle reg.php" method="post"> 

13 


14 <p>Email Address: <input type="text" name="email" size="30" /></p> 
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5 

16 <p>Password: <input type="password" name="password" size="20" /></p> 

J 

18 <p>Confirm Password: <input type="password" name="confirm" size="20" /></p> 

19 

20 <p>Year You Were Born: <input type="text" name="year" 
value="YYYY" size="4" /></p> 

21 

22 <p>Favorite Color: 

23 <select name="color"> 

24 <option value="">Pick One</option> 

25 <option value="red">Red</option> 

26 <option value="yellow">Yellow</option> 

27 <option value="green">Green</option> 

28 <option value="blue">Blue</option> 

29 </select></p> 

30 

31 <p><input type="checkbox" name="terms" value="Yes" /> I agree 
to the terms (whatever they may be) .</p> 

32 

33 <input type="submit" name="submit" value="Register" /> 

34 

35 </form> 

36 

371 /iv 

38 </body> 


39 </html> 
(2) 创建 表单 的 起 始 标 签 : 
<form action= 


如 同 之 前 的 许多 示例 那样 ， 该 页 
被 action 属 性 标识 。 

(3) 为 Email 和 密码 创建 输入 框 : 

<p>Email Address: <input type="text" 


<p>Password: <input type="password" 
<p>Confirm Password: <input type="password" 





"handle_reg.php" method="post"> 








name="email" 
name="password" 





选用 POST 方 法 。 用 来 处 理 的 脚本 是 handle_reg.php， 它 


size="30" /></p> 
size="20" /></p> 
name="confirm" 


size="20" /></p> 


这 些 代 码 行 的 意义 不 言 而 喻 。 /p> 标签 换行 来 增加 Web 浏 览 器 中 的 


间距 。 请 注意 , 创建 了 两 个 密码 输 
框 的 类 型 为 密码 ， 
用 户 再 次 输入 〈 以 便 用 户 对 输入 的 密码 进行 再 次 确认 )。 

(4) 为 生日 创建 输入 


<p>Year You Were Born: 
>value="YYYY" 








第 二 个 输入 框 用 来 确认 第 























[HH 





<input type="text" 
size="4" /></p> 


name="year" 


一 个 框 中 的 输入 内 容 。 输 入 


用户 位 人 风头 (参见 图 6-2) ， 因 此 它 是 一 种 良好 的 保密 方式 ， 让 


这 里 不 使 用 下 拉 菜 单 来 显示 50 年 或 者 100 年 的 选项 ， 而 是 让 用 户 直接 将 他 们 的 出 生年 份 填写 
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在 文本 框 中 。 可 以 在 文本 框 中 预先 设置 value 属 性 来 指明 该 文本 框 所 需 的 正确 年 份 格式 (参见 图 
6-1)。 
(5) 创建 下 拉 菜 单 ， 用 户 可 以 在 下 拉 菜 单 中 选择 喜欢 的 颜色 : 


<p>Favorite Color: 





<select name="color"> 

<option value="">Pick One</option> 
<option value="red">Red</option> 
<option value="yellow">Yellow</option> 
<option value="green">Green</option> 
<option value="blue">Blue</option> 
</select></p> 


事实 上 ,添加 这 个 下 拉 框 的 目的 在 于 ， 它 可 以 在 本 章 后 面 的 一 些 特殊 示例 中 使 用 ,但 是 在 
这 里 它 可 以 用 来 自 定义 用 户 登 录 后 站 点 的 外 观 。 你 可 以 按照 自己 的 意愿 添加 尽 可 能 多 的 颜色 。 
(6) 创建 网 站 条 款 协 议 之 前 的 同意 复 选 框 : 


<p><input type="checkbox"name="terms" Value="Yes" /> 





I agree to the terms (whatever they may be) .</D> 


许多 网 站 一 般 使 用 复 选 框 让 用 户 接受 一 些 条 款 或 许可 协议 。 这 个 表单 没有 连接 到 许可 条 款 页 
面 的 链接 ， 不 过 这 无 关 紧 要 ， 没 有 人 会 去 看 那些 条 款 (不 管 怎么 说 ， 这 个 表单 只 是 一 个 示例 )。 
总 之 ， 这 里 是 要 说 明 PHP 脚 本 是 如 何 处 理 复 选 框 的 。 

0) 添加 一 个 提交 按钮 并 且 关闭 表单 : 


<input type="submit" name="submit" value="Register" /> 
</form> 


(8) 完成 HTML 页 面 : 


</div> 

</body> 

</html> 

(9) 将 文件 保存 为 register .html， 放 在 启用 了 PHP 的 服务 器 上 适当 的 目录 下 ， 并 且 在 Web 
浏览 器 中 加 载 该 页 面 。 

V 提示 

口 注册 页 面 经 常 需要 用 户 对 他 们 的 密码 进行 确认 ， 并 且 有 时 候 还 需要 确认 他 们 的 用 户 名 和 

Email 地 址 (无 论 在 登录 时 需要 什么 信息 )。 

口 大 多 数 注册 页 面 使 用 昵称 或 者 Email 地 址 作为 用 户 名 , 如 果 使 用 Email 地 址 作为 用 户 名 , 这 
将 让 用 户 能 够 更 好 地 记 住 他 们 的 注册 信息 〈 一 个 用 户 可 能 只 有 有 限 几 个 Email 地 址 ， 但 是 
对 于 不 同 的 网 站 可 能 有 好 多 不 同 的 用 户 名 )。 此 外 ,Email 地 址 本 质 上 就 是 唯一 的 ,而 用 户 
名 不 是 唯一 的 。 
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6.2 iE 条件 语句 


编程 基本 的 条 件 语 句 是 标准 的 if 〈 也 曾 被 称 作为 if-then 条 件 语句 ， 现 在 then 被 默认 而 不 
提出 )。 这 种 条 件 语 名 的 语法 非常 简单 


if (condition) { 
statement(s); 





} 

该 条 件 必须 同 圆 括 号 一 同 使 用 , 并 且 语 身 部 分 要 被 放置 在 花 括 号 中 , 这 些 是 需要 执行 的 命令 
(例如 ,打印 字符 串 或 者 将 两 数 相 加 )。 每 句 单独 的 语句 (或 者 命令 ) 都 必须 拥有 自己 的 分 号 以 示 
本 行 的 结束 ， 但 是 对 于 和 一 个 条 件 相关 联 的 语句 数量 并 没有 限制 。 

开发 人 员 通 常 将 这 些 语 句 从 if 起 始 的 代码 行 开 始 进行 缩 进 处 理 ， 用 来 指出 它们 是 条 件 的 结 
果 , 但 是 这 种 格式 并 不 是 语法 上 要 求 的 。 也 有 人 这 样 使 用 语法 : 


if (condition) 


{ 











a 








statement(s); 


} 

如 何 安排 花 括号 是 个 人 喜好 问题 一 一 这 在 网 上 有 一 些小 争议 。 选 择 一 种 你 喜欢 的 风格 ,并 坚 
持 使 用 就 可 以 了 。 

在 每 行 语句 后 没有 使 用 分 号 ， 忘 记 一 个 打开 或 者 关闭 

的 圆 括号 或 者 花 括号 ， 或 者 在 花 括号 后 使 用 了 分 号 都 将 导 人 

致 错误 发 生 。 因 此 在 代码 中 使 用 条 件 语句 时 请 留意 语法 是 
否 正确 。 

PHP 使 用 布尔 值 (TRUE 和 FALSE) 来 检测 是 否 执行 该 
语句 。 如 果 条 件 为 TRUE， 则 执行 语句 ， 如 果 是 FALSE， 则 
不 执行 (参见 图 6-2)。 

通过 本 章 的 学 习 (本 章 的 大 部 分 )， 将 开发 出 一 个 PHP 
脚本 ， 用 来 完全 验证 register .html 表 单数 据 。 在 开始 阶 
段 ， 该 脚本 的 第 一 个 版 本 将 只 创建 这 个 验证 过 程 的 基本 框 图 6-2 IF 条件 语 句 控制 脚本 程序 流 
架 ， 定 义 和 使 用 值 为 布尔 类 型 的 变量 ， 它 将 跟踪 验证 过 程 的 方式 
的 成 败 。 

人 创建 一 个 if 条 件 语 各 


(1) 在 文本 编辑 器 或 者 IDE 中 新 建 一 个 文档 ， 命 名 为 handle_reg .php (参看 脚本 6-2): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Registration</title> 


















TRUE 则 执行 
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</head> 
<body> 
<hl>Registration Results</h1> 


脚本 6-2 将 构建 一 个 PHP 脚 本 的 框架 ， 用 来 验证 表单 数据 


1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 

7 

8 

9 








<title>Registration</title> 
</head> 
<body> 
<hl>Registration Results</h2> 
<?php // 脚本 6-2 - handle_reg.php 
/* 脚本 从 register.html 接 收 8 个 值 : 
email, password, confirm, year, terms, color, submit */ 


// 可 以 在 此 进行 错误 处 理 。 


// 创建 标记 变量 ， 标 记 表 单 状态 : 
17 Sokay = TRUE; 








19 // 如 果 没 有 错误 ， 打 印 一 条 正确 消息 : 
20 if (S$okay) { 


21 print '<p>You have been successfully registered (but not really) .</p>'; 
22 } 

23” ?> 

24 </body> 


25 </html> 
(2) 开始 PHP 脚 本 部 分 ， 如 果 需 要 添加 错误 管理 : 
<?php // 脚本 6-2 - handle_reg.php 


如 果 没 有 开启 display_errors， 或 者 如 果 error_reporting 被 设置 了 错误 的 级 别 ， 请 参 
见 第 3 章 ， 在 这 里 添加 适当 的 代码 以 改变 这 些 设置 。 

(3) 创建 一 个 标记 变量 : 

Sokay = TRUE; 

验证 表单 数据 ， 需 要 使 用 一 个 口 口 变量 ， 用 它 表 示 表 单 是 否 已 经 填写 完毕 。 这 里 称 之 为 “ 标 
记 ” 变 量 是 因为 它 会 存储 一 个 简单 的 值 ， 用 来 指明 事物 的 状态 。 例 如 ，yes 表 示 表 单 填写 完毕 ; 
no 表示 表单 没有 填写 完毕 。 

这 个 变量 被 初始 化 为 布尔 值 中 的 TRUE， 意 思 是 假定 表单 已 正确 填写 完毕 。 布 尔 类 型 不 区 分 
大 小 写 ， 因 此 在 这 里 可 以 书写 为 True 或 者 true。 

(4) 如 果 一 切 就 绪 ， 那 么 打印 一 条 消息 : 


if ($okay) { 
print '<p>You have been successfully registered (but not really) .</p>'; 


} 


HE 
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通过 本 章 的 学 习 ， 验 证 例 程 将 被 添加 进 这 段 脚本 中 ， 用 来 检验 提交 的 表单 数据 。 如 果 有 任何 
数据 在 通过 例 程 检 验 时 失败 ， 那 么 Sokay 将 被 设 定 为 FALSE。 在 这 种 情况 下 ， 该 条 件 语句 的 判断 
结果 也 会 是 FALSE， 因 此 消息 不 会 被 打印 出 来 。 但 是 ， 如 果 数 据 通过 了 每 个 例 程 的 检验 ， 那 么 
$okay 将 为 TRUE， 在 这 种 情况 下 消息 就 会 被 打印 出 来 。 

(5) 完成 PHP 代 码 片 段 和 HTML 页 面 : 




















DP 
</body> 
</html> 


(6) 将 文件 保存 为 handle_reg.php， 放 在 启用 了 PHP 的 服务 器 适当 的 目录 下 ( 同 
register.html 在 同一 目录 下 )， 并 且 在 Web 浏 览 器 中 对 它们 进行 测试 (参见 图 6-3 和 图 6-4)。 

















Please complete this form to register: 





Email Address: 

Password: | .oooeeoe。 
Confirm Password: 

Year You Were Born: wwyy | 


Favorite Color: [Yellow 网) 





园 Iagree to the terms (whatever they may be). Registration Results 


Register ) You have been successfully registered (but not really). 


























图 6-3 ”填写 HTML 表 单 (不 必 填 写 完 整 ) 图 6-4 ”结果 如 此 

当然 , 事实 上 该 脚本 将 总 是 打印 成 功 的 消息 ， 因 为 没有 什么 让 $okay 的 值 为 FALSE。 其 至 可 
以 直接 运行 脚本 并 且 查 看 到 相同 的 结果 。 

V 提示 

口 如 果 条 件 的 语句 段 部 分 只 有 一 行 ， 从 技术 上 来 说 可 以 不 使 用 花 括号 。 这 种 情况 下 ， 可 以 
将 条 件 写 成 以 下 两 种 格式 : 
if (condition) statement; 
或 


if (condition) 
statement; 














可 以 以 这 种 格式 正确 运行 。 但 是 ， 本 书 建议 最 好 一 直 使 用 多 行 加 花 括 号 的 格式 (如 同 在 
语法 介绍 中 示例 的 那样 ) 来 增加 一 致 性 并 减少 错误 。 


6.3 ”验证 函数 


PHP 有 许多 常用 的 函数 来 验证 表单 数据 。 本 章 的 示例 将 用 到 其 中 3 个 重要 的 函数 。 
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第 一 个 是 empty () 函数 , 它 用 来 检验 是 否 一 个 给 定 的 变量 拥有 除 0 和 空 字符 串 之 外 的 其 他 值 。 
当 变量 没有 值 时 〈 或 者 值 为 0 或 者 一 个 空 字符 串 ) ， 它 将 返回 TRUE， 反 之 则 返回 FALSE: 


Svarl = 0; 

$var2 = ee 

Svar3 = /7 一 个 人 
empty(SVar) 7 // RE 没有 已 定义 的 值 
empty a // TRUE， 空 值 


( 
( 
empty ($var2); // TRUE， 非 空 值 
empty ($var3); // TRUE， 空 值 


这 个 函数 用 于 确认 表单 中 的 文本 框 是 否 被 填写 。 例 如 ， 如 果 有 一 个 叫做 Email 的 文本 框 ， 并 
且 用 户 在 提交 表单 前 并 没有 在 其 中 做 任何 输入 ， 那 么 $_POST['email'] 变 量 存在 ,但 是 它 的 值 
是 空 值 。 

下 一 个 是 isset () 函数 ， — ) 几乎 相反 ,虽然 区 别 非常 小 。 当 变量 拥有 值 (包括 0、 
FALSE 或 者 空 字符 串 ) 时 ，isset () 函数 返回 TRUE， 反 之 则 返回 FALSE。 例 如 : 








$varl 0 


$var2 = 'something' 
2 - 和 6 
isset ($var); // 全 没有 已 定义 的 值 

$ 


( 
isset( a // TRUE 
isset ($var2); // TRUE 
isset ($var3); // TRUE 


isset () 函数 通常 用 来 验证 无 文本 的 表单 元 素 ， 如 多 选 钮 、 单 选 钮 和 选择 菜单 。 

最 后 是 is_numeric() 括 ee 反 
之 返回 FALSE。 正 数 、 小 数 ， 甚 至 字符 串 〈 如 果 它 们 是 有 效 的 数字 ) 都 可 以 通过 is_numeric () 
的 验证 : 








$varl = 2309; 
Svar2 = '80.23'; 
Svar3 = 'Bears'; 


is_ numeric($varl); // TRUE 
is_ numeric($var2); // TRUE 
is_numeric($var3); // FALSE 


让 我 们 开始 在 PHP 脚 本 中 应 用 这 些 函 数 ， 以 便 验 证 一 些 实际 数据 。 

过 验证 表单 数据 

(1) 如 果 handle_reg .php 不 在 开启 状态 ， 在 文本 编辑 器 或 者 IDE 中 打开 它 (参看 脚本 6-2 ) 。 
(2) 在 文档 的 head 部 分 ， 定 义 一 个 CSS 类 (参看 脚本 6-3): 


<style type="text/css" media="screen"> 
.error { color: red; } 
</style> 
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脚本 6-3 ”通过 使 用 if 条 件 语句 和 empty() 国 数 ， 这 个 PHP 脚 本 检验 了 是 否 提供 了 Email 地 址 和 
密码 的 值 














工 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5. <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Registration</title> 

7 <style type="text/css" media="screen"> 

8 .error { color: red; } 

9 </style> 

10 </head> 

11 <body> 

12 <hl>Registration Results</h1i> 

13 <?php // 脚本 6-3 - handle_reg.php #2 

14 /* 脚本 从 register.html 接 收 7 个 值 : 

15 email, password, confirm, year, terms, color, submit */ 

16 

17 // 可 以 在 此 进行 错误 处 理 。 

18 

19 // 创建 标记 变量 ， 标 记 表 单 状态 : 

20 Sokay = TRUE; 

21 

22 // 验证 Email 地 址 : 

23 if (empty($_ POST['email'])) { 

24 print '<p class="error">Please enter your email address.</p>'; 
2 $Sokay = FALSE; 

26 } 

2 

28 // 验证 密码 : 

29 if (empty($ POST['password'])) { 

30 print '<p class="error">Please enter your password.</p>'; 

31 $Sokay = FALSE; 

32 } 

33 

34 // 如 果 没 有 错误 ， 打印 一 条 正确 消息 : 

35 if ($okay) { 

36 print '<p>You have been successfully registered (but not really) .</p>'; 
ey 

8 > 

39. </body> 

40 </html> 

这 个 CSS 类 将 被 用 来 格式 化 打印 出 的 注册 错误 信息 。 

(3) 验证 Email 地 址 : 

if (empty($_POST['email'])) { 


print '<p class="error">Please enter your email address.</p>'; 
Sokay = FALSE; 
} 


这 个 if 条 件 语 句 使 用 代码 empty ($_POST['email']) 作 为 它 的 条 件 。 如 果 这 个 变量 是 空 
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的 ， 意 味 着 它 没有 值 、 有 一 个 为 0 的 值 ， 或 者 值 为 空 字符 囊 ， 那 么 条 件 语句 的 结果 是 TRUE。 在 
这 种 情况 下 ， 将 执行 print 语 句 ， 并 且 sokavy 变 量 将 被 研 值 为 FALSE (标志 着 不 是 所 有 的 输入 信 
息 都 正确 )。 

如 果 变 量 不 为 空 , 那么 条 件 语句 的 结果 就 为 FALSE, 将 永远 不 会 调用 print 函 数 ,并且 $okay 
将 保持 它 原 有 的 值 。 

(4) 对 密码 做 同样 的 验证 : 


if (empty($_POST['password'])) { 
print '<p class="error">Please enter your password.</p>'; 
Sokay = FALSE; 

} 


这 是 对 Email 验 证 的 重复 ， 只 有 变量 名 与 print 语 句 进 行 了 相应 的 更 改 。 其 余 的 表单 输入 也 
将 被 及 时 验证 。 

所 有 打印 出 来 的 错误 信息 都 被 放置 在 HTML 中 class 值 为 error 的 <p> 标 签 里 。 这 样 做 是 为 
了 应 用 CSS 的 格式 (也 就 是 说 错误 信息 将 用 红色 的 字体 打印 ， 而 不 是 书 上 显示 的 颜色 (黑白 ))。 

(5) 将 文件 保存 为 handle_reg .php， 并 放置 在 与 register .html 相 同 的 目录 下 (在 启用 了 
PHP 的 服务 器 上 )， 并 且 在 Web 浏 览 器 上 测试 表单 和 脚本 (参见 图 6-5 和 图 6-6)。 


Please complete this form to register: 



































Email Address: 


Password: 


Confirm Password: 0000» 





Year You Were Born: 1901 


Favorite Color: | Green $ Registration Results 
































口 Iagree to the terms (whatever they may be). Please enter your email address 
Register ) Please enter your password 
图 6-5 “如果 遗漏 了 表单 输入 中 的 Email 地 址 或 者 密码 图 6-6 将 看 到 这 样 的 信息 














(6) 再 次 以 不 同 完成 程度 提交 表单 以 测试 更 多 的 结果 。 
如 果 提 供 了 Email 地 址 和 密码 的 值 ， 结 果 将 同 图 6-4 非 常 相似 ， 这 是 因为 Sokay 变 量 的 值 仍然 
为 TRUE。 


Vv 提示 

口 当 在 条 件 语句 中 使 用 函数 时 ， 就 像 这 里 使 用 的 empty () 那样 ， 非 常 容易 因为 遗忘 关闭 圆 

括号 而 看 到 错误 信息 。 请 在 编写 任何 控制 结构 时 对 所 使 用 的 语法 特别 注意 。 

D isset () 函数 的 一 个 用 法 是 用 来 避免 引用 不 存在 的 变量 。 如 果 PHP 被 设置 为 显示 通知 ( 参 
见 第 3 章 )， 那 么 诸如 使 用 未 被 声明 的 变量 $var 将 会 导致 错误 出 现 。 可 以 通过 编写 下 面 的 
代码 避免 这 样 的 错误 发 生 : 


if (isset ($var)) { 
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// 可 以 对 $var 做 任何 想 做 的 事情 。 
} 


口 即便 是 向 PHP 脚 本 发 送 的 所 有 表单 数据 都 使 用 字符 串 的 方式 ,is_numeric () 函数 仍然 能 
够 用 于 表单 中 的 数据 ， 这 是 因为 它 处 理 只 包含 数字 的 字符 串 。 
D isset () 国 数 能 够 接受 任意 数量 的 变量 作为 参数 ， 


if (isset($varl, S$var2)) { 
print 'Both variables exist.'; 


} 


如 果 所 有 被 命名 的 变量 都 被 设置 ,该 函数 将 返回 TURE， 如 果 有 任何 变量 未 被 设置 ， 函 数 
将 返回 FALSE。 


6.4 使 用 else 


在 if 条 件 语 句 之 后 要 介绍 的 下 一 个 逻辑 形式 是 if-else 条 件 语句 。 它 可 以 用 来 构建 这 样 的 条 
件 : 当 条 件 满 足 时 执行 一 组 语句 ,而 条 件 不 满足 

















时 执行 另外 的 语句 : 程序 流 
if (condition) { | 
statement (s); 
} else { 









条 件 为 


other_statement (s); TRUE 则 执行 
生 








} 

使 用 这 个 结构 需要 牢记 的 要 点 是 ， 除 非 完 
全 满足 条 件 , 否则 将 执行 else 语 句 。 换 名 话说 ， 
else 后 的 语句 指明 的 是 默认 行为 ， 因 此 在 if 
条 件 后 的 语句 才 是 规则 的 例外 (参见 图 6-7)。 

让 我 们 重 写 handle_reg.php 页 ， 使 用 
if-else 条 件 语 句 把 对 生日 数据 的 验证 并 入 其 
中 。 在 这 个 过 程 中 ， 将 创建 一 个 新 的 变量 来 表 。 图 6-7 IF-ELSE 条 件 语句 控制 脚本 中 程序 流 的 方式 
示 生 日 。 


信使 用 else 的 步 又 


(1) 如 果 handle_reg .php 不 在 开启 状态 的 话 , 在 文本 编辑 器 或 IDE 中 打开 它 (参看 脚本 6-3)。 
(2) 在 $okay 条 件 之 前 密码 验证 之 后 创建 一 个 新 的 条 件 语句 (参看 脚本 6-4) : 


if (is numeric($_ POST['year'])) { 















































脚本 6-4 添加 if-else 条 件 语 句 后， 可 以 对 生日 的 日 期 进行 验证 并 且 在 这 个 过 程 中 创建 一 个 新 
的 变量 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
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名 <html Xmlns="http://www.w3.org/1999/xhtm1" xml:lang="en" lang="en"> 











4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; 
charset=utf-8"/> 

6 <title>Registration</title> 

学 <style type="text/css" media="screen"> 

8 .error { color: red; } 

9 </style> 

10 </head> 

11 <body> 

12 <hl>Registration Results</h1> 

13 <?php // 脚本 6-4 - handle_reg.php #3 

14 /* 脚本 从 register.html 接 收 7 个 值 : 

15 email, password, confirm, year, terms, color, submit */ 

16 

17 // 可 以 在 此 进行 错误 处 理 。 

18 

19 // 创建 标记 变量 ， 标 记 表 单 状态 : 

20 Sokay = TRUE; 

21 

22 // 验证 Email 地 址 : 

23 if (empty($_POST['email'])) { 

24 print '<p class="error">Please enter your email address.</p>'; 

用 多 Sokay = FALSE; 

26 } 

2 

28 // 验证 密码 : 

29 if (empty($_POST['password'])) { 


print '<p class="error">Please enter your password.</p>'; 
Sokay = FALSE; 


[5 
ODPO 


// 验证 出 生年 : 
if (is numeric($ POST['year'])) { 

$age = 2011 - $_POST['year']; // 计算 年 龄 。 
} else { 


WU WwW WwW 
au 





38 print '<p class="error">Please enter the year you were born as 
four digits.</p>'; 

39 Sokay = FALSE; 

40 } 

41 

42 // 如 果 没 有 错误 ， 打 印 一 条 正确 消息 : 

43 if ($okay) { 

44 print '<p>You have been successfully registered (but not really) .</p>'; 

45 print "<p>You will turn $age this year.</p>"; 

46 } 

47 ?> 

48 </body> 

49 </html> 


由 于 year 变 量 的 值 应 当 是 一 个 数字 ,因此 可 以 使 用 is_numeric() 赫 代 empty () 函数 来 验证 
它 的 值 。 这 只 是 表单 元 素 验 证 的 第 一 步 ， 后 面 的 脚本 会 进一步 验证 表单 元 素 。 
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(3) 创建 一 个 新 的 变量 : 

Sage = 2011 - $_POST['year']; 

如 果 $_POST['year' 
赋值 为 当前 年 份 减 去 用 户 提交 的 年 份 。 由 于 还 没有 介绍 日 
期 函数 ， 这 里 在 等 式 中 硬 编码 当前 年 份 。 

(4) 加 入 一 个 else 子 句 : 


} else { 
print '<p class="error">Please enter 
the year You were born as four digits.</p>'; 
Sokay = FALSE; 
} 


如 果 year 没 有 含有 数值 ， 那 么 将 会 打印 错误 信息 ， 
并 且 $okay 变 量 的 值 将 被 设置 为 FALSE (任何 验证 过 程 失 
败 都 将 是 这 样 的 情况 )。 

(5) 在 最 终 的 print 语 句 后 ，$okay 的 条 件 语句 内 ， 
打印 $age 的 值 : 

print 


如 果 $okay 变 量 的 值 仍 为 TRUE， 那 么 说 明 提交 的 数 








"<p>You will turn S$age this year.</p>"; 



































2 这 意味 着 已 经 计算 过 用 户 的 年 龄 
了 (表明 用 户 今年 有 多 大 岁数 ) ， 并 将 结果 打印 出 来 。 





(6) 将 脚本 保存 在 与 register .htm1 相 同 的 目录 中 





(在 启用 了 PHP 的 服务 器 上 )， 并 在 Web 浏 览 器 上 再 次 测 
试 (参见 图 6-8、 图 6-9 和 图 6-10)。 








V 提示 

口 另外 一 个 好 用 的 验证 函数 是 checkdate() ， 可 以 
用 来 确认 某 个 日 期 是 否 存在 (或 者 在 过 去 是 否 存 
在 )。 可 以 这 样 进行 使 用 : 


if (checkdqate (Smonth，S$dqavy，S$year)) {... 

\ 一 所 pr /rr 

6.5 更 多 运算 符 
前 面 的 章 


(++) 和 自 减 〈--)， 以 及 赋值 运算 符 (=)， 
人 ) ， 它 用 来 将 一 个 字符 串 附 加 给 另外 的 字符 串 。 
齐 符 都 用 来 对 变量 的 值 进行 处 理 , 但 是 


( 
这 些 运算 


] 变 量 含 有 一 个 数值 〈 这 就 意味 着 条 件 为 TRUE )， 


那么 Sage 变 量 将 被 








Please complete this form to register: 





Email Address: me@example.com 
Password: .vv 

Confirm Password: 。。。。 
Year You Were Bom: ywyy | 


Favorite Color: | Green 上 


图 Iagree to the terms (whatever they may be). 





(Register 





市 已 经 对 PHP 中 大 部 分 的 运算 符 以 及 使 用 它们 的 变量 的 类 型 做 了 讨论 
包括 对 数字 的 算术 运算 符 : 加 (+) 减 (-) 乘 (*) 除 (/)， 
用 来 为 变量 设置 值 ， 不 考虑 类 型 。 














图 6-8 再 次 对 表单 进行 测试 ， 
些 年 份 信 息 


忽略 一 











Registration Results 





Please enter the year you were born as four digits. 





图 6-9 ”如 果 年 份 没有 含有 数值 ， 将 打 
印 这 样 的 错误 消息 





Registration Results 


You have been successfully registered (but not really). 





You will turn 57 this year. 


图 6-10 如 果 用 户 正 确 提 交 了 出 生年 
份 (数值 )， 脚 本 会 计算 出 用 
户 的 年 龄 并 打印 出 来 (假设 用 
户 同时 提交 了 Email 地 址 和 密 
码 ) 








值 增加 和 减少 1 的 快捷 方式， 自 增 
我 们 已 经 学 习 了 


它们 在 条 件 语 句 中 却 用 得 比较 少 。 现在 将 开始 
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探索 比较 运算 符 和 逻辑 运算 符 ， 它 们 也 很 重要 。 表 6-1 列 举 了 PHP 中 更 加 全 面 的 运算 符 种 类 。 
表 6-1 PHP 中 的 运算 符 




















运 算 符 用 法 类 型 
+ 加 算术 
z 减 算术 
* 乘 算术 
/ 除 算术 
% 模 (除法 的 余数 ) 算术 
上 + 增 量 算术 
二 减 量 算术 
将 值 赋 给 变量 研 值 
== 相等 比较 
!= 不 相等 比较 
< 小 于 比较 
> 大 于 比较 
<= 小 于 或 等 于 比较 
和 大 于 或 等 于 比较 
! 取 反 逻辑 
AND 与 逻辑 
&& 与 逻辑 
OR 或 逻辑 
| 或 逻辑 
XOR 或 非 逻辑 

连接 字符 串 





6.5.1 ”比较 运算 符 


当 赋值 运算 符 〈 等 号 ) 第 一 次 在 第 2 章 中 介绍 时 ， 我 们 已 经 认识 到 它 的 含义 同 之 前 按照 常规 
理解 的 含义 有 所 不 同 。 代 码 行 





Svar = 5; 
并 不 代表 $var 等 于 5， ee 0 这 是 非常 重要 的 差别 。 
当 编 写 条 件 语 句 时 , 将 经 常 希望 了 解 某 个 变量 是 否 等 于 某 个 特定 的 值 ( 同 用 户 名 或 者 密码 匹 


配 ， 等 等 ) ， 这 就 不 能 只 使 用 等 号 〈 这 是 因为 它 是 用 来 赋值 而 不 是 表示 同 某 值 相等 )。 出 于 这 种 目 
的 ， 需 要 使 用 相等 运算 符 (==)， 它 用 两 个 等 号 来 表示 相等 关系 : 


Svar = 5; 
if. (Svar == 5) { 0. 


行 代码 一 同 使 用 将 首先 把 $var 的 值 设置 为 5， 然 后 当 $var 的 值 等 于 5 时 条 件 的 结果 为 





> 


> 
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TRUE。 这 个 示例 再 次 解释 了 PHP 中 等 号 用 法 的 重要 不 同 ， 以 及 必须 将 赋值 运算 符 和 比较 运算 符 
仔细 区 分 开 的 原因 。 
下 一 个 是 比较 运算 符 : 不 等 于 ， 它 由 一 个 惊叹 号 和 一 个 等 号 组 成 〈(!=)。 此 外 ， 剩 下 的 比较 
运算 符 和 它们 在 数学 中 使 用 的 是 相同 的 : 小 于 (<)、 大 于 (>)、 小 于 等 于 (<=) 和 大 于 等 于 (>=)。 
作为 对 比较 运算 符 的 示范 , 将 对 用 户 的 生日 年 份 是 否 在 2011 年 之 前 进行 确认 , 并 且 还 要 确认 
密码 是 否 同 原始 密码 匹配 。 


党 使 用 比较 运算 符 














(1) 如 果 handle_reg .php 不 在 开启 状态 的 话 ， 在 文本 编辑 器 或 者 IDE 中 打开 它 (参看 脚本 








6-4) 。 
(2) 在 密码 验证 之 后 ， 检 查 两 个 密码 是 否 匹配 (参看 脚本 6-5 ) 
if ($_POST['password'] != $_POST['confirm']) { 


print '<p class="error">Your confirmed password does not 
一 match the original password. </p>'; 
Sokay = FALSE; 

} 





脚本 6-5 ”页 面 处 理 脚 本 的 这 个 版 本 使 用 比较 运算 符 以 进行 对 密码 和 year 值 的 验证 


水 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 
5. <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Registration</title> 

<style type="text/css" media="screen'"> 

8 .error { color: red; } 

9 </style> 

</head> 

<body> 

<hl>Registration Results</h1> 

<?php // 脚本 6-5 - handle_reg.php #4 

/* 脚本 从 register.html 接 收 7 个 值 : 

email, password, confirm, year, terms, color, submit */ 





// 可 以 在 此 进行 错误 处 理 。 


PIANrROGODNPO 





Ko) 


// 创建 标记 变量 ， 标 记 表 单 状态 : 





20 Sokay = TRUE; 

21 

22 // 验证 Email 地 址 : 

23 if (empty($_POST['email'])) { 

24 print '<p class="error">Please enter your email address.</p>'; 
2 Sokay = FALSE; 

26 } 

27 

28 // 验证 密码 : 

29 if (empty($_POST['password'])) { 
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30 print '<p class="error">Please enter your password.</p>'; 

3 Sokay = FALSE; 

325 才 

33 

34 // 检查 两 个 密码 是 否 相 等 : 

35 if ($ POST['password'] != $ POST['confirm']) { 

36 print '<p class="error">Your confirmed password does not match 
the original password.</p>'; 

37 S$Sokay = FALSE; 

38 } 

3:9 

40 // 验证 出 生年 : 


1 if (Is_numeric(S$_POST['year'])) { 

42 Sage = 2011 - $_POST['year']; // Calculate age this year. 

43 } else { 

44 print '<p class="error">Please enter the year you were born as four 
digits.</p>'; 

45 Sokay = FALSE; 

46 } 

47 





48 // 检查 出 生年 是 否 在 当前 年 之 前 : 
49 if ($ POST['year'] >= 2011) { 





50 print '<p class="error">Either you entered your birth year Wrong or 
You come from the future!</p>'; 

51 $Sokay = FALSE; 

52 } 

53 


54 // 如 果 没 有 错误 ， 打 印 一 条 正确 消息 : 
55 if ($okay) { 


56 print '<p>You have been successfully registered (but not really) .</p>'; 
57 print "<p>You will turn Sage this year.</p>"; 

S58 :} 

59 

60 ?> 

61 </body> 

62 </html> 


为 了 比较 这 两 个 字符 串 的 值 , 可 以 使 用 不 等 于 运算 符 。 你 也 可 以 使 用 一 个 字符 串 比 较 函 数 ( 参 
见 第 5 章 )， 不 过 这 已 经 足够 了 。 
(3) 在 year 验 证 之 后 ， 检 查 year 的 值 是 否 在 2011 年 之 前 : 


if ($_POST['year'] >= 2011) { 
print '<p class="error">Either you entered your birth year 
"Wrong or you come from the future!</p>'; 
Sokay = FALSE; 





} 

如 果 用 户 键入 的 出 生年 份 为 2011 或 者 在 其 之 后 , 将 会 有 错误 显示 (可 以 根据 实际 时 间 修 改 年 
份 )。 

(4) 将 脚本 保存 在 与 register .html 相 同 的 目录 中 (在 启用 了 PHP 的 服务 器 上 ), 并 且 在 Web 
浏览 器 中 再 次 测试 (参见 图 6-11 和 图 6-12)。 
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Please complete this form to register: 





Email Address: me@example.com 





PassWoOrd: 。。。oooooosooooooosoooos。 





Confirm Password: .，。 


Year You Were Born: 2023 





Favorite Color: f Green 入 | Registration Results 
图 Iagree to the terms (whatever they may be). 


Your confirmed password does not match the original password 














、Register Either you entered your birth year wrong or you come from the future! 














图 6-11 再 次 运行 表单 图 6-12 ”两 个 新 的 验证 检查 





w 提示 

口 在 比较 两 个 来 自 于 表单 的 字符 串 值 (如 密码 和 密码 确认 ) 之 前 ,建议 对 它们 两 个 使 用 trim() 
国 数 ， 以 去 除 多 余 的 空格 。 本 书 在 这 里 没有 这 样 做 的 原因 是 避免 将 问题 处 理 得 过 于 复杂 ， 
但 是 推荐 保持 这 个 良好 的 习惯 。 

口 另外 一 个 检查 文本 输入 框 已 经 被 填充 的 方法 如 下 所 示 (而 不 是 使 用 empty () 函数 ), 例如 : 
if (strlen ($var) > 0 ) { 

// Svar 正 确 。 
} 

口 在 一 个 i£ 条 件 语句 中 ， 如 果 在 编写 Ssvar == 5 时 错误 地 写 为 Svar = 5， 将 看 到 相应 的 条 
件 语 句 总 是 会 执行 。 这 是 因为 尽管 条 件 $var == 5 也 许 是 或 者 也 许 不 是 TRUE ， 但 条 件 
$var = 5 的 结果 都 将 是 TRUE。 

口 一 些 程序 员 提 倡 反 向 编写 条 件 语句 ， 如 : 
if (5 == Svar) { 
虽然 这 样 看 起 来 很 怪 ， 但 是 当 你 不 经 意 间 编写 出 如 5 = $var 这 样 的 代码 ， 就 会 出 现 一 个 
错误 (更 容易 捕获 这 个 错误 )， 因 为 数字 5 不 能 被 赋值 给 其 他 的 值 。 









































6.5.2 ”逻辑 运算 符 


使 用 PHP 编 写 的 条 件 能 够 对 TRUE 或 者 FALSE 状 态 进 行 识别 。 正 如 已 经 看 到 的 那样 ， 这 可 以 
使 用 函数 和 比较 运算 符 很 好 地 实现 。 逻 辑 运算 符 是 我 们 在 本 章 中 要 讨论 的 最 后 一 种 运算 符 类 型 ， 
它 可 以 用 来 创建 更 多 的 复杂 或 者 显而易见 的 结构 。 

举 一 个 在 PHP 中 关于 TRUE 条 件 的 简单 例子 ， 用 来 确认 一 个 变量 的 值 不 为 零 或 者 空 字符 串 或 
者 FALSE， 代 码 如 下 : 


Svar = 5; 
if ($var) { ... 


我 们 已 经 看 到 在 进行 处 理 的 PHP 脚 本 中 使 用 了 $okay 变 量 。 
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下 面 使 用 逻辑 运算 符 表 达 的 条 件 结果 也 是 TRUE: 

(Ds 

当 引 用 没有 值 的 变量 (或 者 值 为 0 或 一 个 空 字符 串 ) 或 创建 的 结构 不 合 逻 辑 时 ， 条 件 结果 将 
为 FALSE。 下 面 的 条 件 结果 将 是 FALSE: 

计生 (天 -二 二 志和 条 

在 PHP 中 , 感叹 号 (!) 是 NOT 运 算 符 。 可 以 用 来 对 语句 的 TRUE/FALSE 状 态 进 行 转化 。 例 如 : 


Svar = 'value'; 
if ($var) {... // TRUE 





if (!S$var) {... // FALSE 

if (isset(S$var)){... // TRUE 
if (!isset($var)){... // FALSE 
if (!empty($var)){... // TRUE 





除了 这 些 简单 的 部 分 条 件 ，PHP 还 提供 了 5 种 逻辑 运算 符 ， 两 个 版 本 的 与 “AND 和 &&)、 两 个 
版 本 的 或 (OR 和 | | ， 或 使 用 两 个 称 为 管道 的 字符 ) 、 以 及 异 或 〈XOR)。 当 一 个 运算 符 有 两 种 选择 
时 〈 例 如 ， 使 用 与 和 或 时 ) ， 它 们 仅仅 只 有 优先 级 上 的 区 别 。 在 几乎 所 有 的 情况 中 ， 都 可 以 交替 
地 使 用 与 或 者 或 的 任何 一 种 版 本 。 

使 用 圆 括 号 和 逻辑 运算 符 可 以 创建 更 加 复杂 的 if 条 件 语句 。 对 于 一 个 AND 条 件 语句 来 说 , 每 
一 个 组 成 部 分 都 必须 是 TRUE， 这 样 才 能 保证 整个 条 件 的 结果 为 TRUE。 对 于 OR 来 说 ， 至 少 有 一 
个 子 句 是 TRUE 才能 保证 整个 条 件 的 结果 是 TRUE。 以 下 的 条 件 均 为 TRUE: 





if ( (5 <= 3) OR (5 >= 3) ) { ... 

if ( (5>3) AND (5<10) ) {... 
以 下 的 条 件 均 为 FALSE: 

if ( (5 != 5) AND (5>3)){.. 

if ( (5 != 5) OR (5<3)){.. 





在 构造 自己 的 条 件 语句 时 ， 请 记 住 两 件 重要 的 事情 : 首先 , 为 了 让 符合 条 件 结 果 的 语句 能 够 
执行 ， 条 件 的 值 必须 为 TRUE， 第 二 ， 使 用 圆 括号 时 ， 可 以 忽略 优先 级 的 规则 ， 并 强制 运算 符 按 
照 所 制定 的 顺序 进行 处 理 。 

为 了 举例 说 明 逻 辑 运算 符 ， 我 们 向 handle_reg.php 页 面 添加 了 两 个 条 件 语句 ， 还 可 以 将 
yeazr 条 件 语句 内 套 进 其 他 的 条 件 语句 中 (参看 框 注 “ 峰 套 条 件 ” 以 了 解 更 多 内 容 )。 





说 套 条 件 


闻 时 加 加 寺村 加 
加 gg 
加 DDDDOYYDYD 

EC On on 


EO 
statement(s)2; 
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yelse /2 
else 


other_statement(s)2; 
yy O20 
关上 加 提 | 和 国 
other_statement(s)1; 


i 


DOO000000000000000000000000000U00UUPHPIUUUO 
se 
UUOUUUUUiUesisesU00000 











党 使 用 逻辑 运算 符 
(1) 如 果 handle_reg .php 不 在 开启 状态 的 话 ， 在 文本 编辑 器 或 者 IDE 中 打开 它 (参看 脚本 
6-5)。 


(2) 删除 已 经 存在 的 year 验 证 部 分 (参看 脚本 6-6)。 


脚本 6-6 这 里 处 理 的 PHP 脚 本 已 经 被 修改 为 对 于 yeaz 验 证 的 同时 使 用 了 多 个 条 件 语句 和 网 和 套 
条 件 语 句 的 方式 。 同 时 还 会 验证 是 否 勾 选 注册 协议 的 复 选 框 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Registration</title> 
<style type="text/css" media="screen"> 

.error { color: red; } 
</style> 

</head> 

<body> 

<hl>Registration Results</h1i> 

<?php // 脚本 6-6 - handle_reg.php #5 

/* 脚本 从 register.html 接 收 7 个 值 : 

email, password, confirm, year, terms, color, submit */ 


‘OOTPODPpP 





// 可 以 在 此 进行 错误 处 理 。 


IANrRGDNPO 





\O 


// 创建 标记 变量 ， 标 记 表 单 状态 : 
Sokay = TRUE; 


DD DI 
Ky Pe 


// 验证 Email 地 址 : 
if (empty($_ POST['email'])) { 

print '<p class="error">Please enter your email address.</p>'; 
5 Sokay = FALSE; 


DD 
心 
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26 
27 
28 
23 
30 
3 
32 
33 
34 
35 
36 


37 
38 
39 
40 
41 
42 
43 
44 
45 
46 
47 


48 
49 
50 
51 
52 
53 


54 
55 
56 
5 
58 
59 
60 
61 
62 
63 
64 
65 
66 
67 
68 
69 
70 
7 


} 

// 验证 密码 : 

if (empty($_POST['password'])) { 
print '<p class="error">Please enter your password.</p>'; 
Sokay = FALSE; 

} 

// 检查 两 个 密码 是 否 相 等 : 

if ($_POST['password'] != $_POST['confirm']) { 


print '<p class="error">Your confirmed 
password does not match the original 
password.</p>'; 

Sokay = FALSE; 


// 验证 年 : 
if ( is numeric($ POST['year']) AND (strlen($ POST['year']) == 4) ) { 


// 检查 出 生年 是 否 在 2011 年 之 前 。 
if ($_ POST['year'] < 2011) { 
$age = 2011 - $_POST['year']; // 计算 年 龄 。 
} else { 
print '<p class="error">Either you entered your birth year 
‘wrong or you come from the future!</p>'; 
$Sokay = FALSE; 
} // 条 件 2 结 束 。 


} else { // 不 满足 条 件 1， 则 : 
print '<p class="error">Please enter the year you were born as 
four digits.</p>'; 
Sokay = FALSE; 


} // 条 件 1 结束 。 


// 验证 用 户 是 否 选中 条 款 : 
if (!isset($ POST['terms'])) { 


print '<p class="error">You must accept the terms.</p>'; 
S$Sokay = FALSE; 

} 

// 如 果 没 有 错误 ， 打 印 一 条 正确 消息 : 


if (S$okay) { 


print '<p>You have been successfully registered (but not really) .</p>'; 
print "<p>You will turn S$age this year.</p>"; 

} 

?> 

</body> 

</html> 
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我 们 已 经 完整 地 将 这 些 条 件 重 写 为 障 套 条 件 ， 因 此 最 好 是 将 这 些 旧 版 本 全 部 删除 。 

(3) 检查 year 变 量 是 否 为 一 个 4 位 的 数字 : 

if ( is numeric($_ POST['year']) AND (strlen($ POST['year']) == 4) ) { 

这 个 条 件 由 两 部 分 组 成 。 第 一 个 部 分 我 们 已 经 看 到 ， 是 用 来 测试 值 是 否 为 有 效 的 数字 。 第 二 
个 部 分 用 来 获取 year 变 量 的 长 度 (使 用 strlen() 国 数 )， 并 且 检 查 其 值 是 否 为 4。 由 于 使 用 的 是 
aAND 运 算 ， 因 此 条 件 每 个 部 分 的 结果 都 为 TRUE 时 整个 条 件 才能 为 TRUE。 

(4) 创建 一 个 子 条 件 来 验证 year 的 值 不 在 2011 之 后 : 


If ($_POST['year'] < 2011) {S$age = 2011 - $_POST['year']; 
} else { 
print '<p class="error">Either 
”You entered your birth year 
”Wrong or you come from the 
future!</p>'; 
Sokay = FALSE; 
} // 条 件 2 结束 。 


这 个 if-else 条 件 句 在 主 条 件 句 中 充当 语句 部 分 ， 只 有 在 条 件 为 TRUE 时 才 运 行 这 部 分 。 这 
个 if-else 用 来 检查 year 变 量 是 否 大 于 等 于 2011 ( 即 ， 用 户 的 生日 必须 早 于 当前 年 份 ) 。 如 果 条 
件 为 TRUEDU 就 说 明 用 户 年 龄 已 经 计算 完成 ;否则 会 打印 一 条 错误 信息 ， 并 且 $okay 变 量 将 被 设 
置 为 FALSE (指明 有 问题 出 现 )。 

注意 ， 这 个 条 件 语句 与 前 一 个 相反 : 验证 值 小 于 某 个 值 ， 而 不 是 大 于 或 等 于 某 个 值 。 

(5) 完成 year 的 主 条 件 句 : 


} else { // 不 满足 条 件 1， 则 : 
print '<p class="error">Please enter the year you Were born 
"as four digits.</p>'; 
Sokay = FALSE; 
} // 条 件 1 结束 。 


else 子 句 完 成 了 在 第 (3) 步 中 开始 的 条 件 语 句 。 如 果 至 少 有 一 个 条 件 为 FALSE， 那 么 就 会 打 
印信 息 ， 并 且 $okay 被 设置 为 FALSE。 
(6) 确认 用 户 选中 条 款 协议 复 选 框 : 


if (!isset($_POST['terms'])) { 
print '<p class="error">You must accept the terms.</p>'; 
Sokay = FALSE; 

} 


如 果 没 有 设置 $_POST['terms'] 变 量 , 说 明 用 户 没 有 选中 复 选 框 , 那么 就 会 打印 错误 消息 。 
条 件 语句 更 加 准确 的 写法 如 下 : 

If ( !isset($_POST['terms']) AND ($_POST['terms'] == 'Yes') ) { 

0) 这 是 对 脚本 的 唯一 变更 之 处 ,将 脚本 再 次 保存 并 放置 在 与 register .html 相 同 的 目录 
中 (在 启用 了 PHP 的 服务 器 上 )， 并 且 在 Web 浏 览 器 中 再 次 测试 (参见 图 6-13 和 图 6-14)。 

(8) 如 果 需 要 ， 可 以 将 year 的 值 改 成 将 来 的 年 份 并 且 再 次 提交 表单 (参见 图 6-15)。 
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Please complete this form to register: . . 
ee BA Registration Results 


Email Address: me@example.com 


Please enter the year you were born as four digits. 





PassWord: .vv 


You must accept the terms. 











Confirm PassWord: 。。。。 











WN 








6-14 ”如 果 没 有 正确 填写 字段 或 没有 选中 协 
议 复 选 框 ， 将 会 打印 出 错误 信息 





Year You Were Born: 15 








Favorite Color: | Green $ 





















































口 Iagree to the terms (whatever they may be). Registration Results 
( Register ) Either you entered your birth year wrong or you come from the future! 
图 6-13 ”PHP 脚 本 遇 到 不 为 4 位 数字 的 年 份 时 会 图 6-15 ”year 验 证 检查 日 期 是 否 在 2011 年 之 前 
捕获 错误 ， 提 交 本 图 填写 的 表单 就 会 
出 现 这 种 情况 
w 提示 


口 在 本 书 中 一 直 保 持 着 另 一 个 常见 的 编程 惯例 ， 即 书写 术语 TRUE 和 FALSE 时 使 用 全 大 写 方 
式 ， 虽 然 这 不 是 PHP 必 需 的 。 例 如 ， 下 面 的 条 件 是 TRUE: 


tf (Ere (2 


口 在 处 理 长 而 复杂 的 条 件 时 非常 容易 忘记 开启 或 者 关闭 的 圆 括 号 或 花 括 号 ， 这 将 导致 错误 
出 现 或 者 意 想不到 的 结果 发 生 。 寻 找 一 个 有 助 于 让 代码 变 得 清晰 的 方式 (例如 ， 分 
隅 条 件 语 句 并 使 用 注释 )。 另 一 项 实用 的 技术 是 先 创建 完整 的 条 件 结构 ， 然 后 再 添加 细 市 
部 分 。 
口 如 果 执 行 if-else 语 句 时 有 问题 ， 将 变量 的 值 打 印 出 来 将 能 帮助 调试 问题 。 条 件 不 为 
TRUE 时 ， 是 因为 变量 的 值 不 是 预计 的 那样 


6.6 使 用 elseif 


if-elseif (或 者 if-elseif-else) 条 件 同 if-else 条 件 类 似 。 它 的 作用 就 像 是 运行 if 
语句 ， 并 且 能 够 按照 需要 进行 任意 长 度 的 扩展 : 


if (conditioni1) { 
statement(s); 

} elseif (condition2) { 
other_statement(s); 





tt 





o 


} 
是 另外 一 个 示例 (参见 图 6-16): 


if (condition1) { 
statement(s); 
} elseif (condition2) { 





122 060 0000 





other_statement(s); 
} else { 
other_ other_statement(s); 


} 










条 件 为 
TRUE 则 执行 





























图 6-16 ”IF-ELSEIF-ELSE 控 制 脚本 中 程序 流 的 方式 


必须 永远 让 else 位 于 条 件 语 句 的 最 后 一 部 分 ， 因 为 只 有 到 这 里 没有 满足 任何 一 个 条 件 才 会 
执行 它 (或 者 说 ，else 代 表 了 默认 的 行为 )。 但 是 ， 也 可 以 将 elseif 用 作 if 条 件 中 的 一 部 分 并 
且 继 续 多 次 使 用 。 


作为 示例 ， 我 们 创建 一 个 条 件 语句 ， 基 于 所 选 color 值 来 打印 消息 。 
信使 用 elseifE 的 步 又 


(1) 如 果 handle_reg.php 不 在 开启 状态 的 话 ， 在 文本 编辑 器 或 者 IDE 中 打开 它 〈 参 看 脚本 
6-6)。 
(2) 在 $okay 条 件 前 ， 创 建 一 个 新 的 条 件 语句 (参看 脚本 6-7): 


if ($_POST['color'] == 'red') {S$color type = 'primary'; 














脚本 6-7 这 个 多 行 的 if-elseif-else 条 件 打 印 了 关于 color 的 特定 信息 ， 并 且 验 证 提交 的 
color 的 值 是 否 为 允许 的 值 


浊 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
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<html Xmlns="http://www.w3.org/1999/xhtm1" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Registration</title> 
<style type="text/css" media="screen"> 
.error { color: red; } 
</style> 
</head> 
<body> 
<hl>Registration Results</h1> 
<?php // 脚本 6-7 - handle_reg.php #6 
/* 脚本 从 register.html 接 收 7 个 值 : 
email, password, confirm, year, terms, color, submit */ 


// 可 以 在 此 进行 错误 处 理 。 


// 创建 标记 变量 ， 标 记 表 单 状态 : 
Sokay = TRUE; 





// 验证 Email 地 址 : 

if (empty($_POST['email'])) { 
print '<p class="error">Please enter your email address.</p>'; 
$okay = FALSE; 


} 
// 验证 密码 : 
if (empty($_POST['password'])) { 


print '<p class="error">Please enter your password.</p>'; 
Sokay = FALSE; 
} 





// 检查 两 个 密码 是 否 相 等 : 

if ($_POST['password'] != $_POST['confirm']) { 
print '<p class="error">Your confirmed 
password does not match the original 
password.</p>'; 
Sokay = FALSE; 


} 
// 验证 年 : 
If ( is numeric($_ POST['year']) AND (strlen($ POST['year']) == 4) ) { 


// 检查 出 生年 是 否 在 2011 年 之 前 。 
if ($_POST['year'] < 2011) { 
Sage = 2011 - $_POST['year']; // 计算 年 龄 。 
} else { 
print '<p class="error">Either you entered your birth year wrong or 
you come from the future!</p>'; 
Sokay = FALSE; 
】 // 条 件 2 结束 。 








}) else { // 不 满足 条 件 1， 则 : 


print '<p class="error">Please enter the year you were born as four 
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digits.</p>'; 
54 Sokay = FALSE; 
55 
56 } // 条 件 1 结 
57 
58 // 验证 用 户 是 否 选中 条 款 : 
59 if ( !isset($_POST['terms']) AND ($_POST['terms'] == 'Yes') ) { 
60 print '<p class="error">You must accept the terms.</p>'; 
61 Sokay = FALSE; 
2 二 
63 
64 // 验证 颜色 : 
65 if ($_POST[I'color'] == 'red') { 
66 $color type = 'primary'; 
67 } elseif ($ POST['color'] == 'yellow') { 
68 $color type = 'primary'; 
69 } elseif ($ POST['color'] == 'green') { 
70 $color type = 'secondary'; 
71 } elseif ($_ POST['color'] == 'blue') { 
72 $color type = 'primary'; 
73 } else { // 有 问题 ! 
74 print '<p class="error">Please select your favorite color.</p>'; 
了 5 Sokay = FALSE; 
76 } 
入 过 
78 // 如 果 没 有 错误 ， 打 印 一 条 正确 消息 : 
79 if ($okay) { 
80 print '<p>You have been successfully registered (but not really) .</p>'; 
81 print "<p>You will turn S$age this year.</p>"; 
82 print "<p>Your favorite color is a $color type color.</p>"; 
83. , 才 
84 ?> 
85 </body> 
86 </html> 


color 的 值 来 自 于 一 个 下 拉 菜 单 ， 它 有 4 个 可 能 的 选项 : red、yellow、green 和 blue。 这 

条 件 语句 将 打印 出 一 条 信息 ， 重 复 用 户 对 color 的 选择 ， 并 且 对 color 进 行 格 式 化 。 第 一 个 条 
件 榨 验 是 否 i$_POST['color'] 的 值 同 字 符 串 red 相 等 。 

在 条 件 语 句 中 ， 一 定 要 使 用 相等 运算 符 (两 个 等 号 )， 而 不 是 赋值 运算 符 (一 个 等 号 )。 

(3) 为 第 二 个 color 添 加 一 个 elseif 子 句 : 


} elseif ($_POST['color'] == 'yellow') { 


$color_ type = 'primary'; 


这 个 elseif 继 续 完成 在 第 二 步 中 创建 的 主 条 件 语 句 。 该 条 件 本 身 是 对 第 二 步 中 条 件 的 重复 ， 
只 不 过 使 用 了 一 种 新 的 color。 
(4) 为 另外 两 种 颜色 添加 elseif 子 句 : 


} 


} 





elseif ($_POST['color'] == 'green') { 
$color_ type = 'secondary'; 
elseif ($_POST['color'] == 'blue') { 


$color_ type = 'primary'; 
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一 旦 理解 了 主要 的 概念 ， 接 下 来 就 只 是 对 每 种 可 能 的 color 值 进行 重复 编写 elseif 子 句 的 
操作 了 。 
(5) 添加 一 个 else 子 句 : 


} else { 
print '<p class="error">Please select your favorite color.</p>'; 
Sokay = FALSE; 

} 


如 果 用 户 没有 选择 color ， 或 者 处 理 表 单 的 时 候 提 交 了 color 其 他 的 值 (red、yellow、 
green 或 plue 之 外 的 值 )， 这 些 条 件 的 结果 都 将 不 为 TURE， 这 意味 着 这 个 else 子 句 将 起 作用 。 
它 将 打印 一 个 错误 并 且 将 $okay 赋 值 为 FALSE， 提 示 有 问题 发 生 。 

color 的 检查 顺序 无 关 紧 要 ,只 要 将 else 子 句 放 在 最 后 即 可 。 

(6) 在 Sokay 条 件 语句 中 ， 打 印 用 户 喜 欢 的 color 类 型 . 


print "<p>Your favorite color is a S$color_ type color.</p>"; 


0) 将 脚本 保存 在 与 register .html 相 同 的 目录 中 (在 启用 了 PHP 的 服务 器 上 ), 然后 在 Web 
浏览 器 上 用 不 同 的 color 选 项 进行 测试 (参见 图 6-17 和 图 6-18)。 





























Registration Results 





You have been successfully registered (but not really). . . 
Registration Results 


You will turn 12 this year. 
































Your favorite color is a secondary color. Please select your favorite color. 
图 6-17 现在 脚本 打印 出 一 条 信息 ， 告 知 用 户 对 图 6-18 对 color 选 择 失 败 导 致 出 现 
color 的 选择 这 条 错误 信息 
w 提示 


口 有 个 问题 很 多 初学 者 都 不 会 意识 到 , 对 黑客 来 说 , 无 需 使 用 你 准备 好 的 HTML 表 单 就 可 以 
将 数据 提交 到 PHP 脚 本 ， Se 这 相当 容易 就 能 做 到 。 因此 , 验证 是 否 包 含 所 需 变量 (你 
所 设置 的 ) 及 其 类 型 和 值 是 非常 重要 的 。 

口 PHP 也 允许 将 elseif 分 写 为 两 个 单词 ， 如 果 愿 意 的 话 : 


if (condition1) { 
statement(s); 

} else if (condition2) { 
Statement(s)2; 





} 


6.7” switch 条 件 语句 


一 旦 使 用 了 更 长 的 1f-elseif-else 条 件 ， 就 会 发 现 使 用 switch 条 件 能 够 节省 时 间 ， 并 且 
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能 够 让 编码 更 加 清晰 。switch 条 件 只 带 有 一 个 可 能 的 条 件 ， 通 常情 况 下 就 只 有 一 个 变量 : 


Switch ($var) { 

case valuel: 
Statement (s)1; 
break; 

case value2: 
Statement (s)2; 
break; 

default: 
statement (s)3; 
break; 


} 

了 解 switch 条 件 语句 如 何 运行 非常 重要 ， 这 将 决定 你 是 否 能 够 正确 地 使 用 它 。 在 switch 关 
键 词 之 后 , 用 一 对 括号 括 住 一 个 已 定义 的 变量 。 PHP 会 按 顺 序 查 看 每 个 case, 尽力 找到 匹配 的 值 。 
注意 , 字符 串 和 数值 的 使 用 方式 遵循 PHP 规 则 , 数值 不 需要 加 引号 , 字符 串 则 需要 用 引号 括 起 来 。 
在 case value 后 面 是 个 冒号 〈 不 是 分 号 ) ， 接 下 来 是 相关 语句 ， 另 起 一 行 缩 进 。 

一 旦 PHP 发 现 case 同 条 件 变 量 的 值 相 匹 配 , 它 将 继续 执行 后 续 语 句 。 下面 是 需要 重点 关注 的 
地 方 : 一 旦 PHP 找 到 匹配 的 case， 它 将 继续 执行 switch 语 句 ， 直 到 switch 条 件 语 句 的 末尾 ( 关 
闭 的 花 括号 ) 或 者 遇 到 break 语 句 ， 直接 在 此 处 退出 switch 结 构 。 因 此 , 使 用 break 语 句 关闭 每 
个 case 非 常 重要 ,出 于 一 致 性 考虑 ,默认 的 case 也 要 关闭 (参见 框 注 “Break、 Exit、 Die 和 Continue” 
中 对 这 些 关 键 字 的 详细 讨论 )。 

前 面 的 switch 条 件 其 实 是 下 面 代码 的 重 写 : 


if (Svar == valuel) { 
Statement (s)1; 

} elseif ($variable == value2) { 
statement (s)2; 

} else { 
Statement (s)3; 

















J 

因为 switch 条 件 使 用 $var 的 值 作为 它 的 条 件 ， 它 首先 检查 $var 的 值 是 否 等 于 valuel， 
如 果 是 ,那么 执行 statement (s)1,， 如果 不 是 , 它 将 检查 是 否 $Svar 的 值 等 于 value2， 如果 是 ， 
执行 结果 为 statement (s)2。 如 果 这 些 条 件 都 不 满足 ，switch 条 件 的 默认 动作 将 执行 
Statement (s)3,。 

考虑 到 这 一 点 ， 让 我 们 用 switch 重 写 color 条 件 语句 。 

信使 用 switch 条 件 

(1) 如 果 handle_reg .php 不 在 开启 状态 的 话 , 在 文本 编辑 器 或 IDE 中 打开 它 (参看 脚本 6-7)。 

(2) 删除 扩展 的 color 条 件 (参看 脚本 6-8)。 
脚本 6-8 switch 条 件 可 以 简化 复杂 的 i1£-elseif 条 件 


1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
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<html Xmlns="http://www.w3.org/1999/xhtm1" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Registration</title> 
<style type="text/css" media="screen"> 
.error { color: red; } 
</style> 
</head> 
<body> 
<hl>Registration Results</h1> 
<?php // 脚本 6-8 - handle_reg.php #7 
/* 脚本 从 register.html 接 收 7 个 值 : 
email, password, confirm, year, terms, color, submit */ 


// 可 以 在 此 进行 错误 处 理 。 


// 创建 标记 变量 ， 标 记 表 单 状态 : 
Sokay = TRUE; 





// 验证 Email 地 址 : 

if (empty($_POST['email'])) { 
print '<p class="error">Please enter your email address.</p>'; 
$okay = FALSE; 


} 
// 验证 密码 : 
if (empty($_POST['password'])) { 


print '<p class="error">Please enter your password.</p>'; 
Sokay = FALSE; 
} 





// 检查 两 个 密码 是 否 相 等 : 

if ($_POST['password'] != $_POST['confirm']) { 
print '<p class="error">Your confirmed password does not match the original 
password.</p>'; 
Sokay = FALSE; 


} 
// 验证 年 : 
If ( is numeric($_ POST['year']) AND (strlen($_ POST['year']) == 4) ) { 


// 检查 出 生年 是 否 在 2011 年 之 前 。 
if ($_POST['year'] < 2011) { 
Sage = 2011 - $_POST['year']; // 计算 年 龄 。 
} else { 
print '<p class="error">Either you entered your birth year wrong or 
you come from the future!</p>'; 
Sokay = FALSE; 
】 // 条 件 2 结束 。 








} else { // 不 满足 条 件 1， 则 : 


print '<p class="error">Please enter the year you were born as four 
digits.</p>'; 
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54 Sokay = FALSE; 

55 

56 } // 条 件 1 结 

57 

58 // 验证 用 户 是 否 选中 条 款 : 

59 if ( !isset($_POST['terms']) AND ($_POST['terms'] == 'Yes') ) { 
60 print '<p class="error">You must accept the terms.</p>'; 

61 Sokay = FALSE; 

62 * 计 

63 


64 // 验证 颜色 : 
65 Switch ($_ POST['color']) { 


66 case 'red': 

67 $color type = 'primary'; 
68 break; 

69 case 'yellow': 

70 $color type = 'primary'; 
71 break; 

72 Case 'green': 

73 $color type = 'secondary'; 
74 break; 

75 case 'blue': 

76 $color type = 'primary'; 
77 break; 

78 default: 

79 print '<p class="error">Please select your favorite color.</p>'; 
80 S$okay = FALSE; 

81 break; 

82 } // switch 结 束 。 

83 


84 // 如 果 没 有 错误 ， 打 印 一 条 正确 消息 : 
85 if ($okay) { 


86 print '<p>You have been successfully registered (but not really) .</p>'; 
87 print "<p>You will turn S$age this year.</p>"; 

88 print "<p>Your favorite color is a $color type color.</p>"; 

89 J} 

90 ?> 

91 </body> 

92 </html> 

(3) 开始 编写 switch: 

switch ($_POST['color']) { 


如 之 前 提 及 的 ，switch 条 件 语句 只 包含 一 个 条 件 : 一 个 变量 的 名 称 。 在 这 种 情况 下 ， 就 是 
S_POST[I'color ']。 
(4) 创建 第 一 个 case: 


Case 'red': 
$color_ type = 'primary'; 
break; 


第 一 个 case 检 查 是 否 $_POST['color'] 值 为 red, 如 果 是 , 就 执行 与 之 前 相同 的 打印 语句 。 
然后 添加 一 个 break 语 句 以 跳出 switch 语 句 。 
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(5) 为 第 二 个 color 添 加 一 个 case: 


Case 'yellow': 
$color_ type = 'primary'; 
break; 


(6) 为 剩 下 的 color 添 加 case， 


Case 'green': 
$color_ type = 'secondary'; 
break; 

case 'blue': 
$color_ type = 'primary'; 
break; 


(7) 添加 一 个 默认 的 case 并 且 完 成 switch 语 句 : 


default: 
print '<p class="error">Please 
一 Select your favorite 
SCOLOr .</p> ' 
Sokay = FALSE; 
break; 
} // switch 结 


这 个 默认 的 case 等 同 于 用 在 原始 条 件 中 的 else 子 句 。 
(8) 将 脚本 在 ee 保存 (在 启用 了 PHP 的 服务 器 上 ), 并且 在 Web 
浏览 器 中 再 次 测试 (参见 图 6-19 和 图 6-20)。 














Registration Results 





You have been successfully registered (but not really). 


Registration Results 


You will turn 47 this year. 














Your favorite color is a primary color. Please select your favorite color. 











图 6-19 ” 当 用 户 选择 了 一 种 颜色 后 , 该 处 理 图 6-20 ”失败 后 的 结果 
脚本 的 显示 结果 














Vv 提示 

口 在 switch 条 件 中 默认 的 case 并 不 是 必需 的 (可 以 对 它 进行 设置 ， 以 便 在 值 不 与 case 中 
任何 一 个 明确 匹配 时 不 做 任何 响应 ) ， 但 是 如 果 使 用 它 ， 就 要 将 它 作 为 最 后 一 个 case 放 
置 。 

口 如 果 在 switch 条 件 中 使 用 字符 串 作 为 case 的 值 ， 那 么 请 注意 它 是 区 分 大 小 写 的 ,这 就 意 
味 着 在 这 种 情况 下 ，Value 同 value 并 不 匹配 。 
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Break、Exit、Die 和 Continue 


PHPUDOUOUOOOUUOOOOOOUOOUUOUUU0O0OU0O0O0O0O0OO0UO0O0O00D0 
U000UpreaxkDUU switcnUOUUU DereasUUOOO0U0O0D000 switcnd 
加 加 fezse 生 | 因 | 加 各/ 和 回首 

Dbreak UU continvef UDO UU0UU0O 
加 时 加 加 加 加 加 时 加 

exit[] diel] preak] OO UUUU PH 
D00000000000000U00exitd0 odiedUUUUPHPIOUUUOUUUUUO 
DOO000000000000000HIMOUUUUUUUU WedU0UUNUUUD die 
dogogggogogoogogouuooo ad 0 saaee)yg 人 ogoogo 


6.8 ”for 循环 


循环 是 本 章 中 讨论 的 最 后 一 个 控制 结构 。 如 同 之 前 建议 的 那样 ， 可 以 用 循环 来 反复 执行 一 个 
代码 段 。 这 里 可 能 会 希望 多 次 打印 某 个 特定 的 信息 ,或 者 希望 打印 出 数组 中 的 每 个 值 。 对 于 以 上 
的 这 些 情况 ， 以 及 更 多 可 能 的 情况 ， 可 以 使 用 循环 来 实现 (后面 的 示例 将 在 第 7 章 中 诠释 )。 

PHP 提 供 3 种 循环 :for 、while 和 foreach。While 循 环 同 for 非 常 相似 ,但 是 在 数据 库 中 
检索 值 或 者 读 取 文 本 文件 (在 边栏 中 对 此 进行 了 介绍 ) 时 使 用 得 更 加 频繁 。foreach 同 使 用 数组 
有 关 ， 这 将 在 第 7 章 中 介绍 。 

for 循 环 用 来 以 指定 的 次 数 反 复 执行 特定 的 语句 (与 while 不 同 ,，while 会 一 直 运 行 直 到 条 
件 为 FALSE, 它们 相似 但 是 在 概念 上 仍然 有 显著 的 区 别 ) 。 可 以 在 循环 中 使 用 临时 计数 (dummy) 
变量 以 达到 这 个 目的 : 

for (initial expression; condition; closing expression) { 

statement(s); 

















} 

initial expression 被 执行 了 一 次 : 在 第 一 次 调用 循环 的 时 候 ， 然 后 condition 用 来 确 
定 是 否 要 执行 statement。 每 当 condition 为 TRUE， 并 且 statement 执 行 完 之 后 ， 会 执行 
closing expression (参见 图 6-21 ) 。 

这 里 有 一 个 循环 的 简单 示例 ， 它 用 来 打印 从 1 到 10 的 数字 : 


for ($v = 1; $v <= 10; Sv++) { 
print Sv; 





} 
练习 for 循 环 ， 我 们 扩展 注册 表单 ， 让 用 户 输入 完整 的 出 生日 期 。for 循 环 可 以 很 容易 地 在 
HTML 表 单 中 创建 下 拉 莱 单 。 
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初始 化 表达 式 


如 果 为 TRUE 
则 执行 









一 旦 条 件 
为 FALSE， 
退出 循环 











图 6-21 














瀛 编写 一 个 for 循 环 


这 个 流程 图 呈现 了 在 PHP 中 foz 循 环 是 如 何 执行 的 








(1) 如 果 register .html 不 在 开启 状态 的 话 , 在 文本 编辑 器 或 者 IDE 中 打开 它 (参见 脚本 6-1)。 


(2) 删除 原来 的 出 生年 份 提示 和 文本 框 (参看 脚本 6-9)。 
脚本 6-9 ”这 个 脚本 使 用 PHP 来 编写 循环 以 动态 地 创建 nonth 下 拉 菜 单 中 day 的 值 


1 
2 
3 
4 
3 
6 
7 
8 
9 





OJAIJOUNWVOPO 


<!DOCTYPE 





content="text/html; 


html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 





xml:lang="en" lang="en"> 


charset=utf-8"/> 


"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0org/1999/xhtml" 
<head> 
<meta http-equiv="Content-Type" 
<title>Registration Form</title> 
</head> 
<body> 


<!-- 





央 本 6-9 - register.php --> 


<div><p>Please complete this form to register:</p> 


<form action="handle reg.php" method="post"> 


<p>Email Address: 


<p>Password: 


<p>Confirm Password: 


<input type="text" name="email" 


<input type="password" name="password" 


size="30" /></p> 


size="20" /></p> 


<input type="password" name="confirm" 
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size="20" /></p> 


19 
20 <p>Date Of Birth: 
21 <select name="month"> 
22 <option value="">Month</option> 
23 <option value="1">January</option> 
24 <option value="2">February</option> 
25 <option value="3">March</option> 
26 <option value="4">April</option> 
27 <option value="5">May</option> 
28 <option value="6">June</option> 
29 <option value="7">July</option> 
30 <option value="8">August</option> 
31 <option value="9">September</option> 
32 <option value="10">October</option> 
33 <option value="11">November</option> 
34 <option value="12">December</option> 
35 </select> 
36 <select name="day"> 
37 <option value="">Day</option> 
38 <?php // Print out 31 days: 
39 for ($i = 1; $i <= 31; S$Si++) { 
40 print "<option value=\"$i\">$i</option>\n"; 
41 } 
42 Sy 
43 </select> 
44 <input type="text" name="year" value="YYYY" size="4" /></p> 
45 
46 <p>Favorite Color: 
47 <select name="color"> 
48 <option value="">Pick One</option> 
49 <option value="red">Red</option> 
50 <option value="yellow">Yellow</option> 
51 <option value="green">Green</option> 
52 <option value="blue">Blue</option> 
b3 </select></p> 
54 
与 与 <p><input type="checkbox" 
name="terms" value="Yes" /> I agree to 
the terms (whatever they may be) .</p> 
56 
5 了 <input type="submit" name="submit" value="Register" /> 
58 
59 </form> 
60 
61 </div> 
62 </body> 
63 </html> 


将 原来 的 一 个 提示 信息 变 为 3 个 ， 以 表示 完整 的 出 生日 期 月、 日 和 年 。 
(3) 将 出 生日 期 的 提示 放 在 密码 确认 之 后 ， 颜 色 选 择 之 前 ， 添 加 文本 提示 和 月 份 列表 : 


<p>Date Of Birth: 
<select name="month"> 
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<option value="">Month</option> 
<option value="1">January</option> 
<option value="2">February</option> 
<option value="3">March</option> 
<option value="4">April</option> 
<option value="5">May</option> 
<option value="6">June</option> 
<option value="7">July</option> 
<option value="8">August</option> 
<option value="9">September</option> 
<option value="10">October</option> 
<option value="11">November</option> 
<option value="12">December</option> 








</select> 

首先 是 一 个 文本 提示 ， 告 诉 用 户 提 供 完 整 的 出 生日 期 。 接 下 来 是 一 个 选择 菜单 ， 用 于 让 用 户 
选择 出 生 月 份 。 选 项 的 值 是 一 个 数值 ， 而 显示 的 文本 是 字符 串 〈 每 个 月 的 英文 单词 ) 。 

(4) 创建 生日 的 选择 菜单 : 

<select name="day"><option value="">Day</option> 

代码 一 开始 是 select 表 单元 素 , 用 于 让 用 户 选 择 出 生日 期 中 的 月 份 。 这 个 列表 的 值 会 由 PHP 
4 | 
(5) 创建 一 个 新 的 PHP 段 : 
<?php 
因为 PHP 能 够 拘 入 到 HTML 中 ， 可 以 用 它 填充 下 拉 菜 单 。 我 们 从 标准 的 PHP 标 记 开 始 。 
(6) 创建 for 循 环 来 打印 31 天 作为 选择 菜单 的 选项 : 


for ($i = 1; $i <= 31; $i++) { 
print "<option value=\"$i\">$i </option>\n"; 


} 

循环 从 创建 一 个 名 为 Si 的 临时 计数 变量 开始 。 在 第 一 次 使 用 循环 时 ,变量 被 设置 为 1。 然 后 ， 
当 信 小 于 等 于 31 时 ， 执 行 循环 体 。 这 些 循环 体 是 print 行 ， 创 建 诸如 

<option value="1">1</option> 

的 代码 ， 后 面 跟 一 个 换行 符 (用 \n 创 建 ) 。 在 循环 体 执行 之 后 ，$i 变 量 的 值 将 会 增加 1。 然 
后 再 次 检查 条 件 ($i<=31) ， 之 后 重复 该 过 程 。 

(7) 关闭 PHP 代 码 段 : 


这 


(8) 将 文件 保存 为 register .php。 

现在 必须 把 文件 用 .Php 后缀 名 进行 保存 ， 以 便 PHP 代 码 能 够 被 执行 。 

(9) 将 文件 放置 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 ， 并 且 在 Web 浏 览 器 中 测试 (参见 图 
6-22)。 

只 要 这 些 脚 本 与 handle_reg .php 在 同一 个 目录 中 , 你 其 至 可 以 以 纯 HTML 的 形式 填写 并 提 
交 表单 。 
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(10) 如 果 需 要 看 一 下 生成 的 PHP 选 项 ， 可 以 查看 HTML 源 代码 (参见 图 6-23)。 





Please complete this form to register: 


Email Address: | 





Password: 





<option value="11">November</option> 
Confirm Password: <option value="12">December</option> 
</select> 
<select name="day"> 
<option value="">Day</option> 
<option value=" 1">1</option> 
Favorite Color: [ Pick one |™) <option value="2">2</option> 
Eee <option value="3">3</option> 
<option value="4">4</option> 
<option value="5">5</option> 
<option value="6">6</0option> 





Date Of Birth: {Month f(Day MvvyY | 





口 Tagree to the terms (whatever they may be). 

































































CRegister ) <option value="7">7</option> 

图 6-22 ”新 版 本 的 HTML 表 单 ， 有 一 些 自 动 图 6-23 ”如 果 查 看 表单 的 HTML 源 代码 ， 将 看 到 
生成 的 内 容 for 循 环 生成 的 数据 

V 提示 


口 大 家 习惯 上 喜欢 使 用 简单 的 变量 ($i、$j、$k) 作为 for 循 环 中 的 计数 器 。 

口 样 ， 在 编写 while 和 fo 循环 时 ， 如 果 只 有 一 条 语句 ， 可 以 将 所 有 代码 写 
在 一 行 。 但 是 ， 不 推荐 这 么 做 。 

ee ey 可 以 在 循环 中 放置 条 件 语 句 ， 也 可 以 在 条 件 语 句 中 放置 循环 ， 等 

口 请 格外 关注 所 编写 的 循环 条 件 ， 以 便 循环 能 够 在 某 一 点 上 结束 。 否 则 ， 你 会 创 建 一 个 无 
限 循环 ， 脚 本 将 永远 地 执行 下 去 。 


























while 循 环 


DPHPD 30000000000Uwhilied O00U000UUUUUD TRUE whilel 0 
DO0UD0Dftord0 O0000000000000000000000000000FALSE 
while[]] ODOUUUO 


WEC (Con ero 
statement (s); 


for[] whilel] DUDUO0UUDwailed O00000000000U00U00 
加 加 加 四 a5 es 四国 加 加 回回 加 加 加 时 加 加 加 时 加 加 加 时 加 加 加 加 
whilelU0O0O0O0U00000 


Geont 
statement (s); 
J wnt le (Coreyarelie ea) 


加 ef 加 
D000000000000000000000wiied00U000000000 120 0 
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6.9 回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
com/forum/。 














6.9.1 回顾 


口 为 什么 在 注册 过 程 中 让 用 户 再 次 确认 密码 非常 重要 ? 

口 PHP 中 if 条 件 语 句 的 基本 结构 是 什么 ?if-else、if-elseif 和 if-elseif-else 条 件 
语句 的 基本 结构 是 什么 ? 

口 empty () 和 isset () 函数 的 区 别 是 什么 ? 

口 什么 是 赋值 运算 符 ? 什 么 是 相等 运算 符 口 

口 如 果 不 知道 Svar 变 量 的 任何 信息 ， 以 下 条 件 语 句 是 TRUE 还 是 FALSE: 











if (Svar = 'donut') { 
D 以 下 运算 符 的 含义 : 

Pp xx 

> || 

Pp ! 


口 switch 条 件 语句 的 语法 是 什么 ? switch 一 般 用 于 哪些 情况 ? 
口 for 循 环 语句 的 语法 是 什么 ? 


6.9.2 ”实践 


D 查找 PHP 手 册 中 各 种 操作 符 的 用 法 。 

口 重 写 handle_reg .php 脚 本 ,使 用 变量 代替 硬 编码 的 值 获 取 当 前 年 份 。 

口 出 于 调试 目的 ， 在 handle_reg.php 脚 本 的 起 始 处 加 入 一 些 代码 ， 打 印 接收 到 的 变量 能 
值 。 提 示 : 有 两 种 方式 ， 一 种 简单 ， 一 种 复杂 。 

口 重 写 某 一 版 的 handle_reg.php 脚 本 ， 显 示 用 户 选 择 的 最 喜欢 的 颜色 。 提 示 : 可 能 需要 
使 用 CSS 和 字符 串 连 接 。 

口 更 新 handle_reg .php 脚 本 ， 通 过 3 个 表单 元 素 验 证 用 户 的 生日 : month、day 和 year。 
创建 一 个 变量 ， 显 示 用 户 的 生日 ， 格 式 为 : MM/DD/YYYY (需要 使 用 连接 )。 











使 用 数组 








本 章 内 容 

口 什么 是 数组 

口 创建 数组 

口 向 数组 添加 项 

口 访问 数组 元 素 

口 创建 多 维 数 组 

口 排序 数组 

口 字符 串 和 数组 之 间 的 转换 
口 在 表单 中 创建 数组 

口 回顾 和 实践 








接 下 来 将 要 学 习 数 组 ， 也 是 本 书 介 绍 的 最 后 一 种 变量 。 数 组 同 数值 或 者 字符 串 有 着 显著 的 不 
同 ， 并 且 如 果 没 有 掌握 和 理解 它 ， 大 多 数 PHP 编 程 工作 都 将 无 法 实现 。 

因为 数组 独特 的 特性 ， 和 介绍 其 他 变量 类 型 相 比 ， 本 章 将 专门 放 慢 速度 。 从 这 章 开始 将 全 面 
介绍 数组 的 概念 ,以 及 创建 和 使 用 数组 的 基础 知识 。 接 下 来 将 介绍 多 维 数组 和 一 些 数组 相关 的 函 
数 。 本 章 内 容 还 将 包含 字符 串 数组 的 转化 ， 并 解释 如 何在 HIML 表 单 中 创建 数组 。 


7.1 什么 是 数组 


数组 组 成 一 个 复杂 但 是 非常 有 用 的 概念 。 然 而 数值 和 字符 串 都 是 标量 (scalar) 变量 (这 意 
味 着 它们 只 含有 一 个 单独 的 值 )， 数 组 是 组 合 到 一 个 重 写 变量 的 多 个 值 的 集合 。 一 个 数组 能 够 由 
数值 和 /或 字符 串 (和 /或 其 他 数组 ) 组 成 ， 因 此 比 起 简单 的 字符 串 或 者 数字 ， 数 组 能 够 让 一 个 变 
量 以 指数 级 承载 更 多 的 信息 。 例 如 ， 如 果 希 望 使 用 字符 串 来 创建 一 个 杂货 列表 ， 可 能 会 这 样 编写 
代码 : 
































$iteml = 'apples'; 
Sitem2 = 'bananas'; 
Sitem3 = 'oranges'; 


每 增加 一 个 项 目 ， 就 需要 创建 一 个 新 的 字符 串 。 这 样 做 非常 麻烦 ， 并 且 在 查询 整个 列表 或 者 


7.1 000050 137 








代码 中 的 任何 特定 值 时 会 变 得 非常 困难 。 可 以 通过 将 整个 列表 放置 到 一 个 数组 〈 可 以 命名 为 
$items) 中 来 简化 这 个 问题 ， 该 数组 将 包含 所 需要 的 所 有 项 目 (参见 表 7-1)。 
表 7-1 杂货 列表 数组 


项 目 编号 项 目 

















apples 
bananas 


oranges 


使 用 数组 ， 就 可 以 对 列表 进行 添加 、 排 序 、 搜 索 等 操作 。 我 们 先 讲述 数组 的 语法 ， 了 人 解 如 何 
操作 数组 。 


数组 的 语法 规则 


其 他 的 变量 类 型 一 一 数值 和 字符 串 类 型 一 一 都 有 一 个 变量 名 和 相应 的 值 (例如 ，$first_ 
name 可 以 等 于 Larzry)。 数 组 也 有 名 称 ， 并 且 使 用 相同 的 命名 规则 ， 这 些 规则 如 下 所 示 : 
口 以 美元 符号 开头 ; 
口 紧 接 开头 符号 之 后 为 一 个 字母 或 者 下 划 线 ， 
口 以 字符 、 数 值 或 者 下 划 线 的 任意 组 合 结尾 。 

但 是 数组 的 不 同 之 处 在 于 它们 拥有 多 个 元 素 (element) (可 将 表 7-1 中 每 行 的 内 容 看 作为 一 个 
元 素 )。 一 个 元 素 由 一 个 索引 (index) 或 称 为 键 (key) (这 两 个 词 可 以 相互 替换 ) 及 其 值 组 成 。 
在 表 7-1 中 ， 项 目 编号 就 是 键 ， 而 项 目 为 值 。 

数组 的 值 通过 它们 的 索引 被 引用 。 数 组 能 够 用 数值 或 者 字符 串 作为 它 的 键 (或 两 者 同时 )， 
这 取决 于 你 如 何 设置 。 

一 般 来 说 , 使 用 数组 看 起 来 同 使 用 任何 其 他 变量 一 样 ,除了 引用 特殊 值 时 需要 用 方 括号 ([ ]) 
包含 它 的 键 。 因 此 ，$items 代 表 对 整个 数组 的 引用 ， 而 $items [1] 只 是 针对 数组 中 特定 的 元 素 
在 本 示例 中 表示 对 apple 的 引用 )。 























A 





超 全 局 变量 

加 轩 轩 时 0 $s 680 
日 SECOOKIEI SSESSIONI SENV 人 加 加 由 加 四 加 加 四 加 加 回 

SOSE 加 加 加 加 加 环 9ST 加 莉莉 莉 攻 王 王 王 玛 王 天 加 玫 加 一 一 王 是 天 一 环 旺 如 王 王 加 加 
国人 

<input type="text" name="name"/> 

gg 

D000Os$s cruUO0OUSEIUOUUOOOOUUOOOUUOROUOOOOOUDOODO 
$_COoKIED] DQ O00U0cookied0000000$ sessrond 0 000 sessiong [0000000O 
DO0000000Ds$ rNvd s seRveR] 00000000PHPIO00U0U0ODO 


138 0570 0000 


7.2 创建 数组 
创建 数组 正式 的 方法 是 使 用 array () 函数 。 它 的 语法 如 下 所 示 : 


$list = array ('apples', 'bananas', 'oranges'); 

除非 特别 指出 ， 否 则 数组 的 索引 自动 从 0 开始 。 本 示例 并 未 为 元 素 指 定 特定 的 索引 ， 因 此 
apple 作 为 第 一 项 自动 索引 为 0， 而 第 二 项 索引 为 1， 第 三 项 索引 为 2。 

可 以 在 使 用 array () 时 为 索引 赋值 ; 


$list = array (1 => 'apples', 2 => 'bananas', 3 => 'oranges'); 

因为 PHP 在 处 理 脚本 中 的 空白 时 非常 自由 , 所 以 可 以 使 用 下 面 这 种 多 行 的 样式 让 结构 更 加 易 
于 阅读 : 

$list = array ( 

1 => 'apples', 

2 => 'bananas', 


3 => 'oranges' 
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最 后 ,指定 的 索引 值 并 不 必须 为 数值 ， 还 可 以 使 用 单词 。 这 种 索引 的 技术 在 制作 更 加 有 意义 
的 列表 时 非常 实用 。 作 为 示例 ， 可 以 利用 下 面 的 脚本 创建 一 个 数组 来 记录 一 周 中 每 天 供应 的 汤 。 
该 示例 也 将 诠释 如 何 能 够 以 及 不 能 打印 数组 (虽然 已 经 讲 过 ， 但 需要 反复 练习 )。 


人 创建 一 个 数组 




















(1) 在 文本 编辑 器 或 者 IDE 中 创建 一 个 新 的 文档 ， 命 名 为 soups1 .php (参看 脚本 7-1): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type'"content="text/html; charset=utf-8"/> 
<title>No Soup for You!</title> 

</head> 

<body> 

<hl>Mmm...soups</hi1l> 














脚本 7-1 $soups 数 组 包含 3 个 元 素 并 且 使 用 字符 串 作 为 它 的 键 


1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>No Soup for You!</title> 

yA </head> 

8 <body> 

9 <hi>Mmm.. .soups</hi> 

10 <?php // 脚本 7-1 - soups1.php 

11 /* 脚本 创建 并 打印 出 一 个 数组 。*/ 


FF: 


7.2 0000 139 





12 // 可 以 在 此 进行 错误 处 理 。 


14 // 创建 数组 : 

15 S$soups = array ( 

16 'Monday' => 'Clam Chowder', 

17 'Tuesday' => 'White Chicken Chili', 
18 'Wednesday' => 'Vegetarian' 

19 ); 


21 // 尝试 打印 这 个 数组 : 
22 print "<p>$soups</p>"; 


24 // 打印 数组 的 内 容 : 
25 print r ($soups); 
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28 </body> 
29 </html> 


(2) 开始 脚本 中 的 PHP 片 段 ， 如 果 需 要 ， 添 加 错误 处 理 : 

<?php // 脚本 7-1 - soups1.php 

如 果 没 有 开启 aisplay_errors, 或 者 error_reporting 被 设置 为 错误 的 级 别 , 请 参见 第 3 
章 ， 在 这 里 添加 适当 的 代码 以 改变 这 些 设 置 。 

(3) 使 用 array () 函数 创建 一 个 数组 : 


$soups = array (人 

'Monday' => 'Clam Chowder', 
'Tuesday' => 'White Chicken Chili', 
'Wednesday' => 'Vegetarian' 


); 

这 是 在 PHP 中 用 字符 串 作 为 索引 对 数组 进行 初始 化 (创建 和 为 其 赋值 ) 所 使 用 的 正确 格式 。 
由 于 其 键 和 值 都 是 字符 串 类 型 ， 因 此 需要 用 单 引号 将 之 引用 。 

不 要 在 最 后 一 个 数组 元 素 之 后 不 经 意 地 添加 逗号 (索引 为 Wednesday) ,这样 做 会 导致 解析 错 
误 发 生 。 

(4) 尝试 打印 这 个 数组 : 

print "<p>$soups</p>"; 

正如 即将 看 到 的 那样 ,数组 与 其 他 变量 的 不 同 之 处 还 表现 为 它们 不 能 用 打印 其 他 (标量 ) 变 
量 的 方式 进行 打印 。 

(5) 使 用 与 前 述 不 同 的 print_r () 函数 打印 数组 .: 

print_r ($soups); 

在 第 2 章 中 我 们 已 经 学 习 了 如 何 使 用 print_r () 函数 来 显示 任何 变量 的 内 容 和 结构 。 在 这 里 
使 用 它 ， 可 以 看 到 这 个 函数 的 工作 方式 与 print 处 理 数 组 方式 的 不 同 之 处 。 

(6) 结束 PHP 和 HTML 代 码 部 分 : 

















> 


























D> 
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</body> 
</html> 


(7) 将 文档 保存 为 soups1.php， 放 置 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 ， 并 且 在 Web 训 
览 器 中 进行 测试 (参见 图 7-1)。 
记 住 ， 要 通过 URL 来 运行 PHP 脚 本 。 











Mmm...soups 


Array 


Array ( [Monday] => Clam Chowder [Tuesday] => White Chicken 
Chili [Wednesday] => Vegetarian ) 














图 7-1 由 于 数组 和 其 他 变量 类 型 是 结构 上 不 同 的 ， 因 此 打印 数组 的 请 求 会 导致 单词 
“数组 ”的 出 现 〈 以 及 错误 的 出 现 ， 这 取决 于 PHP 的 版 本 及 其 设置 ) 。 另 外 一 方 
面 ，print_r() 函数 用 来 打印 数组 的 内 容 和 结构 




















Vv 提示 

口 索引 从 0 开始 是 PHP 以 及 大 多 数 编程 语言 的 标准 惯例 。 这 种 计数 系统 看 起 来 如 此 不 自然 ， 
但 是 却 需 要 将 之 坚持 下 去 。 可 以 用 下 面 两 种 技术 来 应 对 : 第 一 种 ， 手 动 将 数组 中 所 有 的 
索引 起 始 都 指定 为 1; 第 二 种 ， 忘 记 从 1 开始 计数 的 方式 。 可 以 尝试 用 哪个 方式 更 加 人 简便， 
但 是 大 多 数 程序 员 都 已 经 适应 了 这 种 奇怪 的 结构 。 

口 必须 在 引用 一 个 数组 的 元 素 时 使 用 与 创建 数组 时 使 用 的 相同 的 索引 。 在 $soups 示 例 中 ， 
即使 该 数组 显然 拥有 第 一 个 元 素 (第 一 个 元 素 通常 被 索引 为 数字 0)， 但 是 $soups [0] 并 
没有 值 。 

口 如 果 使 用 array () 函数 来 定义 索引 ， 那 么 可 以 指定 第 一 个 元 素 的 索引 ， 其 他 元 素 的 索引 
将 延续 第 一 个 元 素 的 索引 值 。 例 如 : 
$list = array (1 => 'apples', 'bananas', 'oranges'); 
现在 bananas 的 索引 为 2 而 oranges 的 索引 为 3。 

D range () 国 数 也 可 以 用 来 创建 数组 ， 同 时 将 数组 中 元 素 的 值 限定 于 一 定 范 围 。 下 面 是 这 

样 的 两 个 示例 : 


Sten = range (1, 10); 
Salphabet = range ('a', 'Zz'); 


口 在 PHP 第 5 版 中 ，range () 函数 包含 一 个 可 以 用 来 指定 增 量 的 参数 : 

sevens = range (0, 100, 2); 

口 如 果 在 脚本 中 用 var_aump () 函数 代 趟 print_r(), 它 将 不 仅 显示 数组 的 内 容 , 同时 还 会 
以 更 加 详细 的 格式 呈现 数组 的 结构 (参见 图 7-2)。 

口 键 为 数值 的 数组 被 称 为 索引 (index) 数组 。 如 果 键 为 字符 串 ， 则 被 称 为 关联 (associative) 
数组 。 在 其 他 语言 中 则 称 关 联 数组 为 散 列 表 (hash)。 
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array(3) { ["Monday"]=> string(12) "Clam Chowder" ["Tuesday"]=> 
string(19) "White Chicken Chili" ["Wednesday"]=> string(10) 
"Vegetarian" } 














图 7-2 var_qdump () 函数 (在 脚本 7-1 中 用 来 殖 代 print_r() 函数 ) 显示 了 
在 数组 中 的 元 素数 量 以 及 每 个 字符 串 值 的 长 度 








7.3 向 数组 添加 项 


在 PHP 中 创建 了 一 个 数组 之 后 , 就 可 以 使 用 赋值 运算 符 (等 号 ) 向 这 个 数组 添加 额外 的 元 素 ， 
这 个 方法 同 为 字符 串 或 者 数值 变量 进行 赋值 非常 类 似 。 当 进行 这 样 的 操作 时 ， 既 可 以 为 所 添加 的 
元 素 指定 键 也 可 以 不 指定 键 ， 但 是 在 其 中 任意 一 种 情况 下 ， 都 必须 将 数组 用 方 括号 进行 引用 。 可 
以 编写 下 面 的 代码 向 $1ist 数 组 添加 两 个 项 : 


$slist[ ] 
$list[ ] 


如 果 向 这 个 已 存在 的 数组 添加 元 素 而 没有 指定 键 , 那么 每 个 元 素 的 索引 都 将 延续 已 有 索引 的 
数值 。 假 设 这 个 数组 同 前 述 章节 中 的 数组 一 样 ， 已 存在 索引 1、2、3， 那 么 pears 现 在 的 索引 将 
为 4， 而 tomatoes 则 为 5。 

如 果 确 实 指定 了 索引 , 那么 它 的 值 将 被 指定 为 相应 的 位 置 。 任 何 已 存在 的 值 此 时 都 将 被 覆盖 ， 
就 像 这 样 : 

$list[3] = 'pears'; 

$list[4] = 'tomatoes'; 


现在 ， 在 数组 中 第 四 个 元 素 的 值 为 Lomatoes， 并 且 $1ist 中 没有 元 素 的 值 为 oranges (其 
值 被 bears 和 覆盖 了 )。 出 于 这 一 点 的 考虑 ， 除 非 打 算 要 覆盖 某 些 已 经 存在 的 数据 ， 最 好 不 要 在 向 
数组 添加 值 的 时 候 为 特定 的 键 命名 。 

但 是 ， 如果 数 组 使 用 字符 串 作 为 索引 , 很 可 能 要 指定 键 , 以 确保 不 出 现 字 符 串 和 数值 键 的 奇 
怪 组 合 。 

测试 这 个 过 程 ， 我 们 将 重 写 soups1 .php 以 为 数组 添加 更 多 的 元 素 。 这 里 还 将 会 在 添加 新 元 
素 之 前 和 之 后 打印 数组 中 的 元 素数 量 ， 以 查看 添加 更 多 元 素 造 成 的 区 别 。 

使 用 strlen () 可 以 获取 一 个 字符 串 所 包含 的 字符 数量 ， 从 而 了 解 获知 该 字符 串 的 长 度 。 类 
似 地 ， 可 以 通过 使 用 count () 以 了 解 在 数组 中 的 元 素数 量 : 


Show_many = count ($array); 





'pears'; 
'tomatoes'; 
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unset ($array[4]); 
unset ($array['name']); 


SEE 
Unset (Sstring); 


DUUUDarray UUNUUUrsed 000000000U00UD 


$array = array(); 


可 加 加 





> 向 一 个 数组 添加 元 素 








(1) 如 果 soups1 .php 不 在 开启 状态 的 话 ， 在 文本 编辑 器 或 IDE 中 打开 它 。 
(2) 在 用 array() 对 数组 进行 初始 化 之 后 ， 添 加 下 面 的 脚本 (参看 脚本 7-2， 命 名 为 


soups2 .php): 


$countl1 = count ($soups); 
print "<p>The soups array originally had Scount1 elements.</p>"; 


脚本 7-2 可 以 通过 使 用 赋值 运算 符 为 每 个 元 素 赋 值 的 方式 直接 向 数组 添加 元 素 。count () 函数 
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将 帮助 持续 跟踪 数组 中 包含 的 元 素数 量 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>No Soup for You!</title> 

</head> 

<body> 

<hli>Mmm...soups</h1> 

<?php // 脚本 7-2 - soups2.php 

/* 脚本 创建 并 打印 出 一 个 数组 。*/ 

// 可 以 在 此 进行 错误 处 理 。 


// 创建 数组 : 

$soups = array (人 

'Monday' => 'Clam Chowder', 
'Tuesday' => 'White Chicken Chili', 
'Wednesday' => 'Vegetarian,' 

J 


// 计算 并 打印 数组 中 当前 元 素 个 数 : 
$count1 = count ($soups); 
print "<p>The soups array originally had $count1 elements.</p>"; 


// 向 数组 中 添加 3 个 元 素 : 
$soups ['Thursday'] = 'Chicken Noodle'; 
$soups['Friday'] = 'Tomato'; 
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28 $soups['Saturday'] = 'Cream of Broccoli'; 

29 

30 // 再 次 计算 并 打印 数组 中 元 数 个 数 : 

31 S$count2 = count ($soups); 

32 print "<p>After adding 3 more soups, the array now has $count2 elements.</p>"; 
33 

34 // 打印 数组 内 容 : 

35 ‘BELNt. Yr. “(SSOuUDSs); 


36 
37 ?> 
38 </body> 


39 </html> 

count () 函数 用 来 获知 在 $soups 中 的 元 素数 量 。 将 这 个 值 赋 给 一 个 变量 值 ， 可 以 很 容易 地 
将 结果 打印 出 来 。 

G) 为 数组 再 添加 3 个 元 素 : 


$ssoups['Thursday'] = 'Chicken Noodle'; 
$soups['Friday'] = 'Tomato'; 
$ssoups['Saturday'] 


这 些 代码 又 向 已 有 的 数组 添加 了 3 种 汤 ， 分别 用 Thursday、Friday 和 Saturday 进 行 索引 。 
(4) 重新 对 数组 中 的 元 素数 量 进行 计数 ， 并 且 将 值 打印 出 来 。 


Scount2 = count ($soups); 
print "<p>After adding 3 more soups, the array now has S$count2 elements.</p>"; 


第 二 个 print 调 用 是 对 第 一 个 的 重复 ， 为 了 了 解数 组 现在 所 包含 的 元 素数 量 。 
(5) 删除 : 


print"<p>$soups</p>"; 

我 们 已 经 不 再 需要 该 行 代码 了 ， 因 此 可 以 将 它 删 除 (现在 知道 不 能 这 样 简 单 地 打印 数组 )。 

(6) 将 脚本 保存 为 soups2 .php， 放 置 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 ， 并 在 Web 浏 览 
器 中 进行 测试 (参见 图 7-3)。 





= 'Cream of Broccoli'; 














Mmm...soups 


The soups array originally had 3 elements. 
After adding 3 more soups, the array now has 6 elements. 
Array ( [Monday] => Clam Chowder [Tuesday] => White Chicken Chili 


[Wednesday] => Vegetarian [Thursday] => Chicken Noodle [Friday] => Tomato 
[Saturday] => Cream of Broccoli ) 














| 








7-3 ”确定 向 数组 成 功 添加 元 素 的 直接 方法 是 在 添加 操作 的 
前 后 对 元 素 的 数量 进行 计数 比较 

















Vv 提示 
口 当 直 接 向 数组 添加 元 素 时 需要 非常 小 心 。 正 确 的 处 理 方法 为 : 
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$array[ ] = 'Add This'; 
或 者 

$array[1] = 'Add This'; 
以 及 错误 的 方法 : 


$array = 'Add This'; 
如 果 忘 记 了 使 用 方 括号 ， 那 么 添加 的 值 将 替代 整个 已 有 的 数组 ， 只 留 下 一 个 字符 串 或 者 
数值 。 
口 代码 ($array[ ] ='Value'); 会 创建 变量 $array， 如 果 它 尚 不 存在 的 话 。 
口 在 使 用 这 些 数 组 时 ， 本 书 用 单 引号 来 引用 键 和 值 。 没 有 必要 进行 插值 (就 像 一 个 变量 那 
样 )， 因 此 双 引 号 在 这 里 并 不 是 必须 的 。 但 是 如 果 愿 意 ， 也 可 以 在 这 里 使 用 双 引 号 。 
口 如 果 键 都 是 数值 、 变 量 或 者 常数 (本 书 将 在 第 8 章 中 涉及 )， 那 么 不 用 〈 并 且 事 实 上 不 应 
该 ) 将 之 用 引号 引用 。 例 如 : 


$day = 'Sunday'; 
$ssoups[$day] = 'Mushroom'; 


口 sizeof () 函数 是 count () 的 别名 。 它 也 返回 在 数组 中 的 元 素数 量 。 









































数组 合并 
和 加 加 
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Snew_array = array_ merge ($arrayl, S$array2); 


DUUUUUUUUUD soups2.phpU 0 


SOUDG2 = nprEav 

:Thursdav => "Chicken Noodie™, 
riday = Tomato 

Saturdav => Oream Of Broceoll rs 

路 

$soups = array merge($soups, $soups2); 


加 


SSOUDS = SSOUDS + S30UDS23 


D0 


$soups += $soups2; 


7.4 访问 数组 元 素 
无 论 是 如 何 创建 数组 的 ， 从 数组 中 检索 某 个 特定 元 素 (或 者 值 ) 只 有 一 个 方法 ， 那 就 是 用 它 
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的 索引 。 


print "The first item is $array[0]"; 


如 果 数 组 的 索引 使 用 字符 串 ， 则 需要 用 引号 来 引用 ,那么 必须 调整 索引 使 用 的 引号 ， 这 是 因 
为 它们 会 同 print 语 法 相 冲 突 。 下 面 这 行 代码 将 引发 这 样 的 问题 (参见 图 7-4): 


print "<p>Monday's soup is $soups['Monday'] .</p>"; 


为 了 解决 这 个 问题 ， 我 们 可 以 用 花 括号 将 整个 数组 结构 括 起 来 参见 图 7-5): 


print "<p>Monday's soup is {$soups['Monday']}.</p>"; 


具有 讽刺 意味 的 是 , 数组 能 够 在 一 个 变量 中 存 贮 多 个 值 的 特性 让 它 很 有 用 , 但 是 同时 又 带 来 
其 他 变量 类 型 所 没有 的 限制 : 必须 知道 数组 的 键 才能 够 访问 它 的 元 素 。 如 果 该 数组 被 设 定 为 使 用 
字符 串 作为 键 (如 数组 $soups), 那么 $soups [1] 则 不 指向 任何 值 (参见 图 7-6)。 出 于 这 个 原因 ， 
同时 变量 不 区 分 大 小 写 ，$soups ['monaay '] 变 得 训 无 意义 ， 这 是 因为 Clam Chowder 是 被 索引 
指向 Ssoups['Monday'] 的 。 
































Parse error: syntax error, Unexpected T_ENCAPSED_AND_WHITESPACE,. 
expecting T_STRING or T_VARIABLE orT_ NUM_STRING in /Users 
/larryullman/Sites/phpvqs4/soups2.php on line 37 














图 7-4 ”在 关联 数组 中 用 双 引 号 引用 特定 的 元 素 会 导致 解析 错误 














Monday's soup is Clam Chowder. 








图 7-5 用 花 括号 封装 数组 元 素 引 用 是 解决 解析 错误 的 方法 之 一 














Notice: Undefined offset: 1 in /Users/larryullman/Sites\phpvqs4/soups2.php on line 37 














图 7-6 引用 并 不 存在 的 数组 索引 会 导致 出 现 Undefined offset 或 Undefined Index 通知 信息 
访问 数组 中 所 有 值 的 最 快捷 而 且 最 简单 的 方法 是 使 用 foreach 循 环 。 这 个 循环 结构 将 遍历 数 
组 中 的 每 个 元 素 : 


foreach ($array as Skey => $value) { 
print "<p>Key is Skey. Value is S$value</p>"; 











} 

循环 的 每 次 迭代 ， 都 会 将 当前 数组 元 素 的 键 赋值 给 $Skey 变 量 ， 将 元 素 的 值 赋值 给 $value 变 
注意 ， 这 里 可 以 使 用 任意 变量 : $Sk 和 $v 都 可 以 。 

现在 我 们 可 以 利用 这 些 知识 编写 一 个 新 的 包含 各 种 汤 的 脚本 。 


这 打印 数组 中 的 值 














(1) 在 文本 编辑 器 或 者 IDE 中 新 建 一 个 文档 ， 命 名 为 soups3 .php (参看 脚本 7-3): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
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<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>No Soup for You!</title> 
</head> 
<body> 
<hl>Mmm...soups</hi1l> 


脚本 7-3 ” ”foreach 循环 是 访问 数组 中 每 个 元 素 的 最 简便 的 方法 





2. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
3 <html xmlns="http://www.w3.0o0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>No Soup for You!</title> 

7 </head> 

8 <body> 

9 <hi>Mmm.. .soups</hi> 

10 <?php // 脚本 7-3 - soups3 .php 

11 /* 脚本 创建 并 打印 出 一 个 数组 。*/ 

12 

13 // 可 以 在 此 进行 错误 处 理 。 

14 

15 // 创建 数组 : 

16 Ssoups = array (人 

17 'Monday' => 'Clam Chowder', 

18 'Tuesday' => 'White Chicken Chili', 

19 'Wednesday' => 'Vegetarian', 

20 'Thursday' => 'Chicken Noodle', 

21 'Friday' => 'Tomato', 

22 'Saturday' => 'Cream of Broccoli' 

23. 全 

4 

25 // 打印 数组 中 的 键 和 值 : 

26 foreach ($soups as $day = > $soup) { 

27 print "<p>$day: $soup</p>\n"; 

28 } 

29 

30 ?> 

31 </body> 

32 </html> 





(2) 开启 页 面 中 的 PHP 代 码 片 段 ， 如 果 愿 意 的 话 ， 添 加 错误 管理 : 
<?php // 脚本 7-3 - soups3 .bphp 
(3) 创建 ssoups 数 组 


SSsoups = array ( 

'Monday' => 'Clam Chowder', 
Tuesday' => 'White Chicken Chili', 
'Wednesday' => 'Vegetarian', 
'Thursday' => 'Chicken Noodle', 
"Friday' => "Tomato”, 
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'Saturday' => 'Cream of Broccoli' 


) ; 
在 这 里 一 次 就 创建 了 整个 数组 ， 虽 然 可 以 使 用 同 前 面 脚本 中 相同 的 方式 (逐步 创建 数组 )。 
(4) 创建 一 个 foreach 循 环 以 打印 每 天 提供 的 汤 : 








foreach ($soups as $day => $soup) 

Mmm...soups 
print "<p>$day: $soup</p>\n"; 

} Monday: Clam Chowder 


foreach 循 环 遍历 数组 Ssoups 的 每 个 元 素 ， 将 每 个 索引 赋值 | Tuesday: White Chicken Chili 
给 $qay， 并 将 每 个 值 赋值 给 ssoup。 这 些 值 现在 打印 在 HTML 段 落 | wednesday wesetarian 
标记 中 。 该 print 语 句 包含 一 个 换行 符 (使 用 \n 创 建 的 )， 它 将 对 
页 面 中 的 HTML 源 代码 造成 影响 。 
(5) 结束 PHP 部 分 和 HTML 页 面 : 





Thursday: Chicken Noodle 


Friday: Tomato 





Saturday: Cream of Broccoli 






























































ee 
A 图 7-7 ”为 数组 中 的 每 个 元 
(6) 将 页 面 保存 为 soups3 .php， 并 放置 在 启用 了 PHP 的 服务 器 ds 
上 适当 的 目录 中 ， 然 后 在 Web 浏 览 器 中 进行 测试 参见 图 7-7)。 
Y 提示 脚本 对 每 个 键 和 值 
. a Ce A 行 访问 而 无 需 寻 
口 使 用 数组 的 方式 之 一 就 是 用 赋值 运算 符 将 特定 元 素 的 值 赋 先 对 它们 是 什么 3 
给 单独 的 变量 : 行 了 解 


Stotal = $array[1]; 
这 样 做 可 以 在 保留 数组 原始 值 的 同时 仍然 能 够 将 它 作 为 独立 的 变量 来 操作 。 
口 如 果 只 需要 访问 一 个 数组 的 值 (而 不 是 它 的 键 )， 那 么 可 以 使 用 foreach 结 构 : 
foreach ($array as $value) { 
// 做 任何 想 做 的 事情 。 
} 
口 访问 数组 中 所 有 元 素 的 另外 一 个 方法 是 使 用 for 循 环 : 


for (Sn = 0; Sn < count ($array); Sn++) { 
print "The value is S$array[$n]"; 


} 
口 在 打印 用 字符 串 作为 键 的 数组 的 值 时 ， 请 记 住 用 花 括号 来 避免 错误 发 生 。 这 里 的 示例 使 
用 的 引号 并 没有 问题 ， 因 此 不 需要 花 括号 : 


Sname = trim ($array['name']); 
$total = $_POST['qty'] * 
$_POST['price']; 


口 花 括号 也 可 以 用 来 分 开 变量 起 始 符 和 美元 符号 ， 或 其 他 字符 ， 


print "The total is S${S$total}."; 
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7.5 创建 多 维 数组 


多 维 数组 表现 得 既 简 单 又 复杂 。 它 的 结构 和 概念 有 一 些 难以 掌握 ， 但 是 用 PHP 来 创建 和 访问 
多 维 数组 却 相 当 简 单 。 

多 维 数组 可 以 用 来 包含 比 标准 数组 更 多 的 信息 , 而 这 是 通过 用 其 他 数组 作为 它 的 值 而 不 是 简 
单 的 字符 串 和 数值 作为 值 来 实现 的 。 例 如 : 



































Sfruits = array ('apples', 'bananas', 'oranges'); 
Smeats = array ('steaks', 'hamburgers', 'pork chops'); 
Sgroceries = array ( 

ELUTES. SS ErULTELS:; 


'meats' => $meats, 
'other' => 'peanuts', 
'cash' => 30.00 

) 


数组 $Sgroceries 现 在 由 一 个 字符 串 (peanuts)、 一 个 浮 点 数值 (30.00) 和 2 个 数组 
($fruits 和 $meats) 组 成 。 

指向 多 维 数 组 中 的 某 个 元 素 会 有 一 点 复杂 。 要 点 是 根据 需要 在 方 括号 中 继续 添加 索引 。 
在 这 个 示例 中 ，bananas 位 于 sgroceries['fruits'][1]。 

首先 ， 使 用 ['fruits'] 来 指向 数组 Sgroceries 中 的 元 素 (在 这 种 情况 下 ， 是 一 个 数组 )。 
然后 再 基于 它 的 位 置 指向 数组 中 的 这 个 元 素 : 它 是 第 二 项 ， 因 此 需要 使 用 索引 [11] 。 
在 下 一 个 任务 中 ， 我 们 将 编写 一 个 脚本 来 创建 另外 一 个 多 维 数组 的 示例 。 


信使 用 多 维 数组 

















(1) 在 文本 编辑 器 或 者 IDE 中 开启 一 个 新 的 文档 ， 命 名 为 Dooks .php (参看 脚本 7-4) : 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; 
"Charset=utf-8"/> 
<title>Larry Ullman's Books and Chapters</title> 
</head> 
<body> 
<hil>Some of Larry Ullman's Books</hi1> 














脚本 7-4 多 维 数组 Sbooks 在 一 个 大 变量 中 存储 了 大 量 的 信息 


加 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

多 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

S <meta http-equiv="Content-Type" content="text/html; 
charset=utf-8"/> 
<title>Larry Ullman's Books and Chapters</title> 

</head> 
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8 <body> 
9 <hil>Some of Larry Ullman's Books</hi1> 





10 <?php // 脚本 7-4 - books .php 

11 /* 脚本 创建 并 打印 出 一 个 多 维 数组 。*/ 

12 // 可 以 在 此 进行 错误 处 理 。 

13 

14 // 创建 第 1 个 数组 : 

15 Sphpvgas = array (1 => 'Getting Started with PHP', 'Variables', 'HTML Forms 
and PHP', 'Using Numbers'); 

16 

17 // 创建 第 2 个 数组 : 

18 S$phpadv = array (1 => 'Advanced PHP Techniques', 'Developing Web 
Applications', 'Advanced Database Concepts', 'Security Techniques'); 

19 

20 // 创建 第 3 个 数组 : 

21 S$phpmysql = array (1 => 'Introduction to PHP', 'Programming with PHP'， 
'Creating Dynamic Web Sites', 'Introduction to MySQL'); 

22 


23 // 创建 多 维 数组 : 

24 $books = array ( 

25 'PHP VQS' => $phpvqs, 

26 'PHP Advanced VQP' => $phpadyv, 
27 'PHP and MySQL VQP' => $phpmysql 
28 ); 


29 
30 /7 打印 出 一 些 全 
31 print "<p>The third chapter of my first book is <i>{$books['PHP 


VQS'] [3]}</i>.</p>"; 

32 print "<p>The first chapter of my second book is <i>{$books['PHP 
Advanced VQP'] [1]}</i>.</p>"; 

33 print "<p>The fourth chapter of my fourth book is <i>{$books['PHP 
and MySQL VQP'] [4]}</i>.</p>"; 





35 // 使 用 foreach 遍 历数 组 : 
36 foreach (Sbooks as Skey => Svalue) { 


37 print "<p>$key: S$value</p>\n"; 
38  } 

39 

40 ?> 

41 </body> 


42 </html> 
(2) 创建 初始 PHP 标 签 ， 如 果 需 要 请 添加 错误 管理 : 
<?php // 脚本 7-4 - books .pbhp 


(3) 创建 第 一 个 数组 : 


Spbhpvdads = array (1 => 'Getting Started', 'Variables', 'HTML Forms 
"and PHP', 'Using Numbers'); 


构建 这 个 多 维 数组 ， 我 们 先 创建 3 个 标准 数组 ， 然 后 将 它们 作为 这 个 更 大 数组 的 值 。 该 数组 
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(被 称 为 SsSphpvqs， 是 PHP for the World Wide Web: Visual QuickStart Guide 的 简称 ) 使 用 数值 作为 
键 ， 同 时 使 用 字符 串 作 为 值 。 数 值 从 1 开始 对 应 章节 编号 。 值 则 为 每 章 的 标题 。 





(4) 创建 接 下 来 的 两 个 数组 : 

$sphpadv = array (1 => 'Advanced PHP Techniques', 'Developing Web 
"Applications', 'Advanced Database Concepts', 'Security Techniques'); 
Sphpmysql = array (1 => 'Introduction to PHP', 'Programming with PHP'， 
»!'Creating Dynamic Web Sites', 'Introduction to MySQL'); 





对 于 每 个 数组 ， 出 于 简便 的 考虑 ， 只 添加 本 书 前 4 章 的 信息 。 
(5) 创建 主体 ， 多 维 数组 : 
Sbooks = array (人 
'PHP VQS' => S$phpvgs, 
'PHP 5 Advanced VQP' => S$phpadyv, 
'PHP 6 and MySQL 5 VQOP' => S$Sphpmysql 
7 
数组 $books 是 本 脚本 中 的 主 数组 。 它 使 用 字符 串 作为 键 (本 书 标题 的 缩写 版 ) 并 且 数组 作 
为 值 。 可 以 用 函数 array () 像 创建 其 他 数组 那样 创建 它 。 
(6) 打印 本 书 第 3 章 的 章 名 : 


print "<p>The third chapter of my first book is <i>{$books['PHP 
”VQOS'] [3]1}</i>.</p>"; 


按照 之 前 介绍 的 规则 ， 为 了 访问 所 有 单独 章 的 章 名 ， 我 们 所 需要 做 的 就 是 从 sbooks 开 始 ， 
首先 是 第 一 个 索引 (['PHP Vs '] )， 然 后 是 下 一 个 索引 〈[31] )。 

因为 我 们 将 把 这 些 内 容 放置 在 一 个 print 调 用 中 ， 所 以 需要 用 花 括号 将 整个 结构 引用 ， 以 避 
免 发 生 解 析 错 误 。 

0) 再 打印 两 个 示例 : 


print "<p>The first chapter of my second book is <1>{Sbooks 
»['PHP Advanced VQP'] [1]}</i>.</p>"; 

print "<p>The fourth chapter of my fourth book is <i>{$books 
>['PHP and MySQL VQOP'][4]}</i>.</p>"; 


(8) 通过 foreach 循 环 来 思 历 数组 Sbooks 以 查看 结果 : 


foreach (Sbooks as SKkey => Svalue) { 
print "<p>$key: S$value</p>\n"; 
} 


变量 $key 存 储 了 书 中 每 个 标题 的 缩写 ， 并 且 $value 变 量 包含 每 个 章节 数 台 
(9) 结束 PHP 代 码 片段 并 且 完 成 HTML 页 面 : 


站 
</body> 
</html> 


(10) 将 文件 保存 为 books .php， 并 保存 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 ， 然 后 在 Web 浏 
览 器 中 进行 测试 (参见 图 7-8)。 














了 
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Some of Larry Ullman's Books 


The third chapter of my first book is HTML Forms and PHP. 

The first chapter of my second book is Advanced PHP Techniques. 
The fourth chapter of my fourth book is Introduction to MySQL. 
PHP VQS: Array 


PHP Advanced VQP: Array 








PHP and MySQL VQP: Array 





图 7-8 开始 的 3 行 由 print 语 名 生成。 最 后 3 行 显 示 了 foreach 循 环 的 结果 
(以 及 因为 试图 打印 数组 而 产生 的 提示 ) 





Vv 提示 
口 为 了 访问 每 个 数组 中 的 每 个 元 素 , 可 以 将 两 个 foreach 循 环 妊 套 使 用 , 如 下 (参见 图 7-9): 


foreach (Sbooks as $title => $chapters) { 
print "<p>s$title"; 
foreach ($chapters as Snumber => $chapter) { 
print "<br/>Chapter Snumber is S$chapter"; 
} 
Brint </BS > 
} 


口 使 用 函数 print_r() 或 者 函数 var_qdump () (将 函数 放置 在 HTML 中 的 <pre> 标 签 范 目 
可 以 获得 更 好 的 显示 格式 ) 可 以 对 整个 多 维 数 组 进行 查看 (参见 图 7-10)。 



























































Array 
{ 
[PHP VQS] => Array 
( 
7 [1] => Getting Started with PHP 
PHP \ QS ; 和 [2] => Variables 
Chapter 1 is Getting Started with PHP [3] => HTML Forms and PHP 
Chapter 2 is Variables [4] => Using Numbers 
Chapter 3 is HTML Forms and PHP ) 
Chapter 4 is Using Numbers [PHP Advanced VOP] => Array 
( 
7 a [1] => Advanced PHP Techniques 
PHP Advanced VQP [2] => Developing Web Applications 
Chapter 1 is Advanced PHP Techniques [3] => Advanced Database Concepts 
Chapter 2 is Developing Web Applications [4] => Security Techniques 
Chapter 3 is Advanced Database Concepts ) 
Chapter 4 is Security Techniques [PHP and MYySOL VOP] => Array 
( 
+ [1] => Introduction to PHP 
PHP and MySQL VQP [2] => Programming with PHP 
Chapter 1 is Introduction to PHP [3] => Creating Dynamic Web Sites 
Chapter 2 is Programming with PHP [4] => Introduction to MYSOL 
Chapter 3 is Creating Dynamic Web Sites ) 
Chapter 4 is Introduction to MySQL ) 
图 7-9 ”一 个 foreach 循 环 包 含 在 另外 一 个 图 7-10 ”函数 var_dump () 显示 了 数组 Sbooks 
foreach 循 环 中 ， 能 够 实现 对 一 个 的 结构 和 内 容 


二 维 数组 中 每 个 元 素 的 访问 
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口 可 以 使 用 一 系列 嵌 套 的 array () 调用 的 方式 在 一 个 语句 中 创建 多 维 数组 (用 以 代 禁 本 示 
例 中 使 用 多 个 步骤 的 方式 )。 但 是 ， 并 不 推荐 这 样 做 ， 这 是 因为 随 着 语句 中 租 套 数量 的 增 
加 ， 引 发 语法 错误 的 可 能 性 就 越 大 。 

口 尽管 本 示例 中 的 所 有 子 数 组 都 有 相同 的 结构 〈 用 数值 作为 索引 ， 并 且 有 4 个 元 素 ) ， 但 这 

并 不 是 多 维 数组 所 必须 的 。 

口 为 了 了 解 关 于 “Larry Ullman Collection” 的 更 多 信息 ,以 及 这 里 涉及 的 三 本 书 , 请 访问 本 
书 的 站 点 : www.LarryUllman.com 。 


7.6 ”数组 排序 


PHP 提 供 多 种 排序 数组 的 方法 (如 果 被 排序 的 值 是 字符 串 ， 可 以 使 用 字母 顺序 排序 法 ， 如 果 
被 排序 的 值 是 数值 ， 则 可 以 按照 数值 方式 进行 排序 )。 当 排序 一 个 数组 时 ， 必 须 牢 记 数 组 是 由 键 
一 值 对 组 成 的 。 因 此 , 一 个 数组 能 够 基于 键 或 者 值 来 进行 排序 。 在 依照 值 进行 排序 的 同时 保持 相 
应 的 键 与 之 对 应 是 非常 复杂 的 ， 或 者 也 可 以 对 值 进行 排序 ， 然 后 再 为 它们 赋予 新 的 键 。 

可 以 使 用 sort () 来 对 值 进行 排序 而 不 用 考虑 键 的 问题 ， 而 使 用 rsort () 可 以 对 这 些 值 进行 
反 向 排序 《同样 不 用 考虑 键 )。 每 种 排序 国 数 的 语法 如 下 : 

function ($array); 


因 些 ，sort () 和 rsort () 的 用 法 如 下 : 


sort ($array); 
rsort ($array); 


可 以 使 用 asort () 来 实现 在 对 值 进行 排序 的 同时 保持 每 个 值 同 它 键 之 间 的 对 应 关系 , 而 使 用 
arsort () 来 实现 在 对 值 进行 反 向 排序 的 同时 保持 每 个 值 同 它 键 之 间 的 对 应 关系 。 

可 以 使 用 ksort () 来 实现 在 对 键 进行 排序 的 同时 保持 每 个 键 同 它 值 之 间 的 对 应 关系 , 而 使 用 
krsort () 来 实现 在 对 键 进行 反 向 排序 的 同时 保持 每 个 键 同 它 值 之 间 的 对 应 关系 。 表 7-2 列 举 了 这 




















表 7-2 ”数组 排序 函数 





函 数 排序 依据 是 否 保持 键 - 值 对 应 关系 
Sort(} Values No 

rsort() Values (inverse) No 

asort() Values Yes 
arsort() Values (inverse) Yes 

ksort() Keys Yes 
krsort() Keys (inverse) Yes 





最 后 ，shuffle() 用 来 对 数组 中 的 元 素 顺序 进行 随机 重组 。 
下 面 是 数组 排序 的 示例 , 这 里 将 创建 一 个 列表 , 其 中 包含 学 生 的 名 称 以 及 他 们 在 测验 中 获得 
的 成 绩 ， 然 后 按照 成 绩 和 姓名 来 排序 这 个 列表 。 
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> 对 数组 进行 排序 
(1) 在 文本 编辑 器 或 者 IDE 中 新 建 一 个 文档 ， 命 名 为 sort .php (参看 脚本 7-5): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; 
charset=utf-8"/> 
<title>My Little Gradebook 
</title> 
</head> 
<body> 











脚本 7-5 PHP 为 数组 排序 提供 多 种 不 同 的 函数 ， 其 中 包括 (这 里 使 用 的 ) arsort () 和 ksort () 














水 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 

6 <title>My Little Gradebook</title> 

7 </head> 

8 <body> 

9 <?php // 脚本 7-5 - sort.php 

10 /* 脚本 创建 、 排 序 并 打印 出 一 个 数组 。*/ 

11 

12 // 可 以 在 此 进行 错误 处 理 。 

13 

14 // 创建 数组 : 

15 Sgrades = array( 

46 "Richard' esS-95. 

17 'Sherwood' => 82， 

18 “Toni 2 98.; 

19 urans™ mS 7 

20 'Melissa' => 75, 

21 'Roddy' => 85 

22 ); 

23 

24 // 以 原始 顺序 打印 数组 : 

25 print '<p>Originally the array looks like this: <br/>'; 

26 foreach (S$grades as S$student => S$grade) { 

27 print "$student: S$grade<br/>\n"; 

28 |} 

29 print '</p>'; 

30 

31 // 个 序数 组 并 打印 。 

32 arsort ($grades); 

33 print '<p>After sorting the array by value using arsort(), the array looks 
like this: <br/>'; 

34 foreach (S$grades as S$student => S$grade) { 

3 print "$student: S$grade<br/>\n"; 


(CD 
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37 print '</p>'; 


39 // 以 键 排序 数组 并 打印 。 

40 ksort ($grades); 

41 print '<p>After sorting the array by key using ksort(), the array looks like 
thiesn. <be/S, 

2 foreach (S$grades as S$student => S$grade) { 
3 print "$student: S$grade<br/>\n"; 
4 } 

5 PEINt VS/ 
6 

2 

8 


?> 
</body> 
9 </html> 


(2) 开始 PHP 代 码 部 分 ， 如 果 需 要 请 添加 错误 管理 : 
<?php // 脚本 7-5 - sort.php 
(3) 创建 数组 : 


$sgrades = arrayl 
'Richard' => 95., 
'Sherwood' => 82, 
"Toni ‘=> 798 
"Pranz' = 87 
'Melissa' => 75, 
'Roddy' => 85 
) 


数组 $grades 由 6 个 学 生 的 姓名 以 及 他 们 相应 的 成 绩 组 成 。 因 为 成 绩 是 数值 类 型 的 值 ， 因 此 
它们 不 需要 在 赋值 时 被 引号 引用 。 
(4) 打印 一 个 标题 ， 然 后 用 foreach 循 环 打 印 出 数组 中 的 每 个 元 素 : 


print '<p>Originally the array looks like this: <br/>'; 
foreach (S$grades as S$student => S$grade) { 
print "$student: S$grade<br/>\n"; 

} 

DITt. He/ 

$grades 数 组 会 打印 三 次 ， 指 示 数 组 状态 的 标题 行 对 我 们 来 说 很 有 用 。 首 先 ， 脚 本 按照 原始 
顺序 打印 数组 。 可 以 用 一 个 foreach 循 环 来 做 到 ， 此 时 每 个 索引 (学 生 的 姓名 ) 都 被 赋值 给 
$student， 并且 每 个 值 (学 生 的 成 绩 ) 都 被 赋值 给 sgrade。 最 后 的 print 调 用 将 关闭 HIML 段 


落 。 

















(5) 将 数组 基于 值 进行 倒序 排列 以 查看 谁 获 得 了 最 高 的 成 绩 : 
arsort ($grades); 
因为 要 了 解 谁 获得 了 最 高 的 成 绩 ， 需 要 使 用 arsort () 而 不 是 asort () 。 后 者 按照 数值 升序 排 
序数 组 ， 将 会 对 评分 进行 如 下 排列 :75、82、85 等 ， 而 不 是 按照 期 望 的 98、95、87 方 式 。 
还 必须 使 用 arsort () 而 不 是 rsort () ,用 于 维护 键 一 值 关 系 (rsort () 会 忽略 与 评级 相关 联 
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(6) 用 另外 一 个 循环 再 次 打印 数组 ( 带 有 标题 ): 


print '<p>After sorting the array by value using arsort(), the array 
—>looks like this: <br/>'; 
foreach (S$grades as S$student => S$grade) { 
print "$student: S$grade<br/>\n"; 
} 
print '</p>'; 


(0) 基于 键 对 数组 进行 排序 以 让 数组 中 学 生 的 姓名 依照 字母 顺序 进行 排列 : 

ksort ($grades); 

函数 ksort () 对 数组 依照 键 进行 了 重组 (在 当前 情况 下 为 字母 顺序 ) 的 同时 保持 了 键 - 值 的 
(8) 最 后 一 次 打印 标题 和 数组 : 


print '<p>After sorting the array by key using ksort(), the array 
looks like this: <br/>'; 
foreach (S$grades as S$student => S$grade) { 
print "$student: S$grade<br/>\n"; 
Drintb v/s 


(9) 用 标准 的 PHP 和 HTML 标 签 完成 脚本 : 


a 
</html> 
(10) 将 脚本 保存 为 sort .php, 放置 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 , 并 在 Web 训 览 器 上 

进行 测试 (参见 图 7-11)。 











Originally the array looks like this: 
Richard: 95 

Sherwood: 82 

Toni: 98 

Franz: 87 

Melissa: 75 

Roddy: 85 


After sorting the array by value using arsort(), the array looks like this: 
Toni: 98 

Richard: 95 

Franz: 87 

Roddy: 85 

Sherwood: 82 

Melissa: 75 


After sorting the array by key using ksort()., the array looks like this: 
Franz: 87 

Melissa: 75 

Richard: 95 

Roddy: 85 

Sherwood: 82 

Toni: 98 











图 7-11 ”可 以 对 数组 进行 各 种 排序 得 到 不 同 的 结果 。 请 在 选择 排序 函数 时 
密切 留意 对 是 否 希 望 保持 键 - 值 的 对 应 关系 
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V 提示 

口 数组 $grades 可 以 创建 为 用 成 绩 作 为 键 、 学 生 姓名 作为 值 的 方式 。 两 种 方式 都 可 行 。 

口 函数 natsort () 和 natcasesort () 排 序 字符 串 (同时 保持 键 一 值 对 应 关系 ) 时 使 用 自然 
顺序 (natural order) 的 方式 。 自 然 顺序 排列 方式 最 显而易见 的 示例 就 是 它 会 将 name2 排 
在 name12 之 前 ， 反 之 sort () 处 理 的 结果 是 name12 在 name2 之 前 。 

口 函数 usort () 、uasort () 和 ursort () 可 以 对 数组 使 用 用 户 自 定义 的 比较 函数 方式 进行 
排序 。 这 些 函数 通常 都 应 用 在 多 维 数 组 中 。 


7.7 字符 串 和 数组 之 间 的 转换 


现在 我 们 已 经 对 字符 串 和 数组 都 有 所 了 解 , 本 节 将 介绍 在 这 两 种 形式 之 间 进 行 转换 的 两 个 函 
。 第 一 个 是 implode () ， 它 用 来 将 数组 转换 为 字符 串 ， 第 二 个 是 exploaqae () 用 来 做 相反 的 处 





























并 六 


以 下 是 使 用 这 些 函 数 的 原因 。 
口 将 数组 转换 为 字符 串 是 为 了 能 够 将 值 附加 在 URL 上 进行 传递 (而 不 能 用 数组 如 此 轻松 地 
实现 )。 
口 将 数组 转换 为 字符 串 是 为 了 将 信息 储存 在 数据 库 中 。 
口 将 字符 串 转 换 为 数组 是 为 了 将 一 个 用 逗号 分 隔 的 文本 区 (例如 表单 中 的 关键 字 搜 索 区 域 ) 
转化 为 相互 独立 的 形式 。 
使 用 explode () 的 语法 如 下 : 
Sarray = explode($separator, $string); 
separator 分 隔 符 指明 了 一 个 或 多 个 字符 ,用 来 区 分 一 个 值 的 结束 和 另外 一 个 值 开始 。 通 常 
情况 下 分 隔 符 是 一 个 喜 号 、 一 个 制 表 符 或 者 一 个 空格 。 因 此 代码 可 能 如 下 所 示 : 


Sarray = explode(',', S$string); 




















或 

$array = explode(' ', $string); 

为 了 将 一 个 数组 转换 为 字符 串 ， 需 要 定义 用 什么 作为 分 隔 符 〈 也 就 是 之 间 的 连接 符 )，PHP 
将 处 理 剩 下 的 工作 : 


$string = implode($glue, Sarray); 
sstring = implode(',', S$array); 





或 

$string = implode(' ', $array); 

下 面 诠释 如 何 使 用 explode () 和 implode () ,这 里 将 创建 一 个 HTML 表 单 用 以 包含 一 个 以 空 
格 进行 分 隔 的 代表 用 户 姓 名 的 字符 串 〈 参 见 图 7-12)。PHP 脚 本 将 会 把 字符 串 转 换 为 数组 以 便 能 够 
对 列表 进行 排序 。 最 后 ， 代 码 将 创建 并 返回 被 按照 字母 顺序 排列 的 字符 串 〈 参 见 图 7-13 ) 。 
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Enter the words you want alphabetized with each individual word separated by a space: 





Brian Sommar Eric Mark Shauna Allison Mike ¢ Alphabetize! ) 




















图 7-12 这 个 HTML 表 单 有 一 个 词语 列表 ， 它 被 list .php 脚 本 按照 字母 顺序 进行 排序 


tT 





An alphabetized version of your list is: 
Allison 

Brian 

Eric 

Mark 

Mike 

Shauna 

Sommar 











图 7-13 ”这 是 一 个 相同 的 列表 ， 按 照 字 母 顺序 对 用 户 进行 排序 。 处 理 过 程 很 迅速 并 且 
很 容易 编写 代码 ， 但 是 如 果 没 有 数组 将 无 法 这 样 轻 松 实现 


之 创建 HTML 表 单 

(1) 在 文本 编辑 器 或 者 IDE 中 开启 一 个 新 的 文档 ,命名 为 list .html 〈 参 看 脚本 7-6 ) : 

<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>I Must Sort This Out! </title> 

</head> 


<body> 
<1-- 脚本 7-6 - list.html --> 

















脚本 7-6 ”这 是 一 个 简单 的 HTML 表 单 ， 用 户 可 以 通过 它 来 提交 一 个 词语 列表 。 其 中 还 包含 对 应 
当 如 何在 谨慎 的 Web 设 计策 略 中 使 用 表单 进行 了 详细 的 介绍 











1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>I Must Sort This Out!</title> 

7 </head> 

8 <body> 

9 <!-- 脚本 7-6 - list.html --> 





10 <div><p>Enter the words you want alphabetized with each individual word 
Separated by a space:</p> 


<form action="list.php" method="post"> 


<input type="text" name="words" size="60" /> 
<input type="submit" name="submit" value="Alphabetize!" /> 





OO 心 w IN FF 
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17 </form> 
18 </div> 
19 </body> 
20 </html> 


(2) 创建 一 个 有 文本 输入 框 的 HTML 表 单 : 


<div><p>Enter the words you want alphabetized with each individual 
”WOrd separated by a space:</p> 

<form action="list.php" method="post"> 
<input type="text" name="words" size="60" /> 


在 这 种 情况 下 对 用 户 做 出 提示 是 很 重要 的 。 例如， 如 果 他 们 输入 一 个 辟 号 分 隔 的 列表 ,这样 
并 不 能 正确 地 处 理 这 个 字符 串 (在 完成 了 两 段 脚 本 之 后 ,尝试 使 用 去 号 而 不 是 空格 进行 分 隔 ， 看 
看 会 发 生 什 么 )。 

(3) 创建 一 个 提交 按钮 ， 并 且 结 束 表单 和 HTML 页 面 : 


<input type="submit" name="submit" value="Alphabetize!" /> 
</form> 
</div> 
</body> 
</html> 


(4) 将 脚本 保存 为 1ist .html， 并 放置 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 。 
现在 我 们 将 编写 1ist .php 页 面 来 处 理由 1ist .html 生 成 的 数据 。 


> 在 字符 串 和 数组 中 进行 转换 
































(GD 在 文本 编辑 器 或 者 IDE 中 开启 一 个 新 的 文档 ， 命 名 为 1ist .php (参看 脚本 7-7): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>I Have This Sorted Out </title> 

</head> 

<body> 

<?php // 脚本 7-7 - list.php 











脚本 7-7 因为 函数 explode () 和 implode() 是 如 此 简单 并 且 强 大 ,你 可 以 快速 并 容易 地 通过 编 
写 儿 行 代码 来 对 提交 的 (几乎 任何 长 度 ) 词语 列表 进行 排序 


1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0o0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>I Have This Sorted Out</title> 

7 </head> 

8 <body> 

9 ”<?php // 脚本 7-7 - list.php 

10 /* 脚本 在 $_POST['words'] 接 收 一 个 字符 囊 。 
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11 接 下 来 将 字符 串 转 换 为 数组 ， 按 字母 顺序 排序 数组 并 打印 出 来 。*/ 
// 可 以 在 此 进行 错误 处 理 。 

es 将 引入 的 字符 串 转 换 为 数组 : 

16 $words array = explode(' ' , $ POST ['words']); 
1 剧 序 数组 : 





19 sort(S$words array); 





21 // 将 数组 转换 为 字符 囊 : 


22 $string words = implode('<br/>', $words array); 


24 // 打印 结果 : 
25 print "<p>An alphabetized version of your list is: <br/>$string words</p>"; 


2 a 
28 </body> 
29 </html> 


(2) 将 引入 的 字符 串 $_POST['words'] 转 换 为 数组 : 

Swords_array = explode(' ' , $_POST['words']); 

该 行 代码 依照 字符 串 $_POST[ 'words'] 创 建 了 一 个 新 的 数组 : $words_array。 在 $_POST 
['words'] 中 词语 和 词语 之 间 的 空格 指明 下 一 个 词语 为 一 个 新 的 数组 元 素 。 因 此 第 一 个 词语 为 
$Swords_array[0] ， 然 后 在 $_POST['words'] 中 有 一 个 空格 ， 接 着 第 二 个 词语 为 
$words_array[1] ， 依 此 类 推 ， 直 到 $_POST['words'] 的 结尾 。 

(3) 对 数组 按照 字母 顺序 进行 排序 : 

sort ($words_array); 

因为 不 需要 保持 $words_array 中 的 键 - 值 对 应 关系 ， 因 此 这 里 可 以 用 sort () 替代 之 前 使 用 
的 asort ()。 

(4) 依照 排序 后 的 数组 创建 新 的 字符 串 : 

$string_words = implode('<br/>', S$words_array); 

数组 不 能 像 字 符 串 那样 很 容易 被 打印 处 理 ， 因 此 需要 将 数组 Swords_array 转 换 为 字符 串 
S$string_words 。 结 果 字 符 串 以 Swordqs_array[0] 的 值 开 头 ， 接 下 来 为 HrML<br/> 标 签 、 
$words_array[1] 的 值 ， 依 此 类 推 。 使 用 <br/> 来 检 代 空格 或 者 去 号 将 会 使 在 浏览 器 中 打印 出 
来 的 格式 更 易于 阅读 。 

(5) 在 浏览 器 中 打印 新 的 字符 串 : 

print "<p>An alphabetized version of your list is: <br/> $string words</p>"; 


(6) 结束 PHP 代 码 片 段 和 HTML 页 面 : 


?> 
</body> 
</html> 
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(7) 将 页 面 保存 为 list .php， 放置 在 和 1ist .html 相 同 的 目录 中 ,并 且 在 Web 浏 览 器 中 对 两 
个 脚本 进行 测试 (参见 图 7-12 和 图 7-13)。 


w 提示 





D 也 可 以 运行 用 join () 函数 编写 的 代码 ， 它 同 implode () 是 同 义 的 。 
7.8 在 表单 中 创建 数组 


通过 本 章 的 学 习 ， 我 们 已 经 在 PHP 页 面 中 完整 地 创建 了 数组 。 然 而 ， 可 以 通过 一 个 HTML 表 
单 向 PHP 脚 本 发 送 数组 的 数据 。 事 实 上 ， 每 当 使 用 $_PosT 时 就 是 这 种 情况 。 但 是 可 以 通过 在 一 
个 HTML 表 单 中 创建 数组 的 方式 来 做 更 多 事情 ， 而 它 将 是 更 庞大 的 $_POST 数 组 中 的 一 部 分 ( 因 
此 $_POST 应 为 多 维 数组 )。 

对 于 这 项 功能 ,最 符合 逻辑 的 使 用 就 是 对 于 复 选 框 的 处 理 , 用户 可 以 选择 多 个 相关 的 选项 ( 参 
见 图 7-14)。 复 选 框 的 HTML 源 代码 如 下 : 


<input type="checkbox" 
name="topping" value="Ham" /> 





问题 是 为 



































Pizza Toppings: 口 Extra Tomato DO Ham O Sausage OO Pepperoni 
口 Black Olives OO Turnips 口 Kumquats 








图 7-14 














在 HIML 表 单 中 的 复 选 框 ， 显 示 多 个 可 选项 








了 向 PHP 脚 本 发 送 多 个 值 ， 每 个 表单 元 素 必 须 有 一 个 唯一 的 名 称 。 如 果 创 建 了 多 个 

复 选 框 ， 每 个 的 名 称 都 是 topping， 那 么 只 有 最 后 一 个 复 选 框 的 值 将 被 PHP 脚 本 获取 。 如 果 为 每 

个 复 选 框 (Ham、Tomato、Black-Olives 等 ) 都 创建 一 个 唯一 的 值 ， 会 是 一 件 非常 烦琐 的 工作 。 
解决 这 个 问题 的 方法 就 是 使 用 数组 ， 如 下 面 的 例子 所 示 。 


这 在 HTML 表 单 中 创建 数组 











(GD 在 文本 编辑 器 或 者 IDE 中 开启 一 个 新 的 文档 ， 命 名 为 event .html (参看 脚本 7-8): 


<lDOCTYP 








E html PUBLIC 


"-//W3C//DTD XHTML 1.0 Transitional//EN" 








"http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 





<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Add an Event</title> 

</head> 

<body> 

<!-- 脚本 7-8 - event.html --> 


<div><p>Use this form to add an event:</p> 


脚本 7-8 ”该 HTML 表 单 用 一 个 数组 来 处 理 多 选 框 的 名 称 


1 <!DOCTYPE html PUBLIC 


"-//W3C//DTD XHTML 1.0 Transitional//EN" 
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2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
6 <title>Add an Event</title> 

7 </head> 

8 <body> 

9 <!-- 脚本 7-8 - event.html --> 

10 <div><p>Use this form to add an event:</p> 

11 

12 <form action="event.php" method="post"> 

上 

14 <p>Event Name: <input type="text" name="name" size="30" /></p> 

15 <p>Event Days: 

16 <input type="checkbox" name="weekdays[ ]" value="Sunday" /> Sun 
17 <input type="checkbox" name="weekdays[ ]" value="Monday" /> Mon 
18 <input type="checkbox" name="weekdays[ ]" value="Tuesday" /> Tue 
19 <input type="checkbox" name="weekdays[ ]" value="Wednesday" /> Wed 
20 <input type="checkbox" name="weekdays[ ]" value="Thursday" /> Thu 
21 <input type="checkbox" name="weekdays[ ]" value="Friday" /> Fri 
22 <input type="checkbox" name="weekdays[ ]" value="Saturday" /> Sat 
23 </p> 

24 <input type="submit" name="submit" value="Add the Event!" /> 

25 

26 </form> 

27 </div> 

28 </body> 


29 </html> 
(2) 开始 HTML 表 单 部 分 : 

<form action="event.php" method="post"> 

这 个 表单 会 提交 到 与 这 个 HIML 页面 位 于 同一 个 文件 夹 下 的 event .php 脚 本 。 
(3) 为 事件 名 称 创建 一 个 文本 输入 框 : 

<p>Event Name: <input type="text" name="name" size="30" /></p> 

这 个 示例 让 用 户 能 够 输入 事件 的 名 称 和 事件 发 生 在 周 几 。 

(4) 创建 一 周 七 天 的 复 选 框 : 

















<p>Event Days : 

<input type="checkbox" name="weekdays[ ]" value="Sunday" /> Sun 
<input type="checkbox" name="weekdays[ ]" value="Monday" /> Mon 
<input type="checkbox" name="weekdays[ ]" value="Tuesday" /> Tue 
<input type="checkbox" name="weekdays[ ]" value="Wednesday" /> Wed 
<input type="checkbox" name="weekdays[ ]" value="Thursday" /> Thu 
<input type="checkbox" name="weekdays![ ]" value="Friday" /> Fri 
<input type="checkbox" name="weekdays[ ]" value="Saturday" 

/> Sat 

D> 


所 有 这 些 复 选 框 都 使 用 aays [] 作为 name 的 值 , 这 也 在 PHP 脚 本 中 创建 了 一 个 $_POSTI[ 'days'] 
数组 。 每 个 复 选 框 中 的 value 属 性 是 不 同 的 ， 分 别 对 应 一 周 的 七 天 。 
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(5) 完成 HTML 表 单 : 

<input type="submit" name="submit" value="Add the Event!" /> 
</form> 

(6) 完成 HTML 页 面 : 

</div> 

</body> 

</html> 


(7) 将 页 面 保存 为 event .html， 并 放置 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 。 
我 们 还 需要 编写 一 个 event .php 页 面 来 处 理 这 个 HTML 表 单 。 


过 处 理 HTML 表 单 











(1) 在 文本 编辑 器 或 者 IDE 中 开启 一 个 新 的 文档 ， 命 名 为 event .php (参看 脚本 7-9): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Add an Event</title> 

</head> 

<body> 




















脚本 7-9 ”该 PHP 脚 本 获取 到 了 由 $_POST['days' ] 的 值 构 成 的 数组 


1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0o0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; 
charset=utf-8"/> 








6 <title>Add an Event</title> 

7 </head> 

8 <body> 

9 <?php // 脚本 7-9 - event.php 

10 /* 脚本 处 理 HTMIL 表 单 。*/ 

11 

12 // 可 以 在 此 进行 错误 处 理 。 

13 

14 // 打印 文本 : 

15 print "<p>You want to add an event called <b>{$_POST['name']}</b> which 
takes place on: <br/>"; 

16 

17 // 条 件 语句 ， 检验 是 否 选中 选项 : 

18 if (isset($ POST['days']) AND is array($ POST['days'])) { 

19 

20 foreach ($_ POST['days'] as $day) { 

21 print "$day<br/>\n"; 

22 } 


78 UUUUOUDUUD 163 





24 } else { 
25 print 'Please select at least one weekday for this event!'; 
.6:3 省 


28 // 完成 这 个 段落 : 
29 print '</p>'; 
30, 3 

31 </body> 

32 </html> 


(2) 创建 初始 的 PHP 标 签 ， 添 加 错误 管理 (如果 需要 的 话 )， 并 且 打 印 一 个 介绍 性 的 消息 : 


<?php // 脚本 7-9 - event.php 
print "<p>You want to add an event called <b>{$_POST['name']}</b> 
一 Which takes place on: <br/>"; 


print 行 打印 出 事件 名 称 的 值 。 在 该 脚本 的 实用 版 本 中 , 应 该 添加 一 个 条 件 用 来 验证 是 否 输 
入 了 值 (参见 第 6 童 )。 

(3) 添加 一 个 条 件 语 句 用 来 检验 至 少 有 一 个 选项 被 选中 : 

if (isset($_POST['days']) AND is array($_POST['days'])) { 

如 果 所 有 复 选 框 都 没有 被 点 击 , 那么 $_POST[' days' 1 将 没有 值 存在 。 为 了 避免 由 于 引用 一 
个 不 存在 的 值 而 引发 的 错误 ， 条 件 的 第 一 个 部 分 将 用 来 检验 $_POST[' days' 1] 是否 被 设置 。 

条 件 的 第 二 个 部 分 一 一 确认 $_POST['days'] 为 一 个 数组 ， 两 部 分 条 件 都 为 TRUE 时 ， 整 个 
条 件 才 为 TRUE。 这 一 步 非 常 必要 ， 这 是 因为 当 foreach 循 环 获 得 的 变量 不 是 数组 的 时 候 将 会 产 
生 错 误 (参见 图 7-15)。 





























Warning: Invalid argument supplied for foreach() in /Users 
/larryullman/Sites/phpvgs4/event.php on line 16 














图 7-15 ”由 于 foreach 循 环 的 变量 不 是 数组 而 产生 的 常见 问题 
(4) 打印 每 个 选中 的 选项 : 


foreach ($_POST['days'] as $day) { 
print "$day<br/>\n"; 


可 以 通过 对 数组 $_PosT ['dqays ' ] 运行 Eoreach 循 环 进行 志 历 的 方式 打印 出 每 个 所 选 的 选 





项 。 该 数组 包含 每 个 被 选中 的 复 选 框 中 的 值 (来 自 于 HTML 表 单 的 输入 ， 如 Monday、Tuesday， 
等 等 ) 。 

(5) 完成 is_array() 条 件 : 

} else { 


print 'Please select at least one weekday for this event!'; 


} 
如 果 没 有 选项 被 选中 ， 那 么 isset () AND is_array() 条 件 将 为 FALSE， 并 且 会 打印 这 条 
消息 。 
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(6) 完成 主要 段落 部 分 、PHP 代 码 部 分 以 及 HTML 页 面 : 


BELTAt, /DY 
?> 

</body> 
</html> 


(7) 将 页 面 保存 为 event .php, 放置 在 同 event .ntml 相 同 的 目录 中 , 并 且 在 Web 浏 览 器 中 进 
行 测试 (参见 图 7-16、 图 7-17 和 图 7-18)。 



































Ue EE MS On You want to add an event called Training Seminar which takes 
Event Name: Training Seminar place on: 
Tuesday 
Event Days: O Sun O Mon Tue 园 Wed Thu O Fri O Sat Wednesday 
Add the Event! Thursday 
图 7-16 ”有 复 选 框 的 HTML 表 单 图 7-17 HTML 表 单 的 处 理 结果 


























You want to add an event called Training Seminar which takes 
place on: 
Please select at least one weekday for this event! 




















图 7-18 ”如 果 用 户 没 有 选择 任何 复 选 框 ， 将 看 到 如 图 所 示 的 信息 








Vv 提示 

口 这 里 诠释 的 技术 同样 能 够 让 用 户 用 来 对 下 拉 菜 单 进行 多 项 选择 。 只 需要 为 菜单 使 用 诸如 
something[l] 的 语法 进行 命名 即 可 ,然后 PHP 脚 本 将 获得 $8_POST[' something'] 中 的 每 
个 选项 。 





1Iist() 函 数 
iis€E 人 是 上 回回 加 O000000o0o00oo 


sdate = array ("Thursday', 23. ‘October'ys 
list($weekday, S$day, $month) = $date; 


D0 Sweekday] DO 0 Thursday] OO 0 $a9ay UO O00 230 smonth0 (000O 


October[| 
000oiisd)0000000000isd)0 O0000000o0o0gogggoggog 
国 ste)ODUODODVUOTUOTUVOoo 虹 VOODOOOO 


list ($Sweekday, , S$month) = $date; 


D0 


Ist (montn 三 省 SS 


UOUOUUU00D0 
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list($weekday, $month) = $date; 


i468 欠 人 轩辕 加 加 加 有 gO 人 O000 


7.9 回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
comy/forumy/ 。 














7.9.1 回顾 


口 索引 数组 与 关联 数组 之 间 的 区 别 是 什么 ? 

口 什么 时 候 应 该 用 引号 将 数组 的 索引 和 值 括 起 来 ?什么 时 候 不 应 该 用 ? 

口 如 何 打印 数组 中 的 某 个 元 素 ?” 如 何 打印 数组 中 的 所 有 元 素 ? 

口 如 果 在 添加 数组 元 素 时 不 使 用 方 括号 ， 会 发 生 什么 ? 

口 哪个 函数 返回 数组 中 元 素 的 个 数 ? 

口 打印 数组 元 素 时 ， 什 么 时 候 必须 用 花 括号 括 起 来 ? 

口 sort () 和 asort () 函数 的 区 别 是 什么 ”sort () 和 rsort () 函数 的 区 别 是 什么 ? 

口 explode () 国 数 的 语法 是 什么 ? implode () 函数 的 语法 是 什么 ”如 果 你 忘记 了 ， 请 查看 
PHP 手 册 。 























7.9.2 ”实践 
D 查找 PHP 手 册 中 与 数组 有 关 的 函数 。 研 究 一 下 其 他 常用 的 数组 国 数 。 我 建议 你 熟悉 一 下 


array_key_exists()、array_search() 和 in_array ()。 

口 重 写 soups2 .php， 显 示 数 组 元 素 的 个 数 ， 这 次 不 要 使 用 单独 的 变量 。 提 示 : 在 print 语 
名 中 串联 count () 函数 。 

口 创建 一 个 脚本 ， 建 立 并 显示 一 个 多 维 数 组 〈 或 任意 一 种 数组 )。 

口 重 写 1ist.php， 用 foreach 代 替 imploaqe () ,但 依旧 在 浏览 器 中 按 顺序 每 行 打印 一 个 单 
词 。 加 入 表单 验证 功能 ， 以 便 只 在 存在 实际 字符 串 值 的 时 候 对 其 进行 解析 和 排序 。 








创建 Web 应 用 程序 








本 章 内 容 

口 创建 模板 

口 使 用 外 部 文件 
口 使 用 常量 

口 使 用 日 期 和 时 间 
口 再 谈 使 用 PHP 处 理 HIML 表 单 
口 使 表单 更 具 粘 性 
口 发 送 Email 

口 输出 缓冲 

口 处 理 HITP 头 

口 回顾 和 实践 


本 书 已 经 介绍 了 很 多 PHP 编 程 的 基础 知识 ,现在 就 可 以 将 这 些 技术 组 合 到 一 起 ， 尝 试 创建 一 
个 实际 的 Web 应 用 程序 。 本 章 将 讨论 更 多 函数 和 技巧 ， 让 Web 站 点 更 加 专业 、 功 能 更 丰富 并 且 更 
易于 维护 。 

首先 , 将 讨论 如 何 使 用 外 部 文件 将 Web 页 面 划分 为 独立 的 部 分 (在 一 定 程度 上 能 够 将 逻辑 和 
格式 分 开 )。 然 后 , 介绍 PHP 中 的 常量 和 特殊 数据 类 型 。 此 处 , 我 们 还 将 介绍 PHP 的 一 些 与 日 斯 和 
时 间 相 关 的 函数 。 

本 章 中 有 两 个 主题 用 来 讨论 非常 实用 的 技术 : 使 用 同一 个 页 面 同时 显示 和 处 理 HTML 表 单 ， 
以 及 让 表单 记 住 用 户 提交 的 值 。 此 后 ， 还 将 看 到 PHP 能 够 非常 容易 地 发 送 电 子 邮 件 。 最 后 ， 本 章 
还 将 讨论 更 为 高 级 的 主题 ， 即 输出 缓冲 和 使 用 HTTP 头 。 


8.1 创建 模板 
迄今 为 止 , 每 一 个 示例 都 有 一 页 脚本 专门 处 理 HTML 表 单 、 对 数组 进行 排序 或 执行 运算 。 然 


而 ， 在 开始 开发 多 页 面 的 Web 站 点 ( 称 作 Web 应 用 程序 ) 时 ， 在 大 量 的 页 面 中 重复 常见 的 元 素 就 
会 变 得 不 切实 际 。 
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在 更 复杂 的 Web 站 点 中 ， 有 些 特性 (如 HTML 设 计 ) 会 在 站 点 中 的 每 个 页 面 里 都 用 到 。 可 以 
这 些 元 素 放 到 每 一 个 独立 的 页 面 中 , 但 当 需 要 进行 修改 时 ， 又 必须 一 次 又 一 次 地 完成 修改 。 创 


建 模板 可 以 将 重复 的 内 容 和 特定 于 页 面 的 素材 分 开 ， 
点 可 能 会 拥有 导航 、 版 权 和 其 他 特性 ， 


这 样 就 节省 了 不 少时 间 。 例 如 ， 某 个 Web 站 
这 些 特 性 在 多 个 页 面 中 重复 使 用 (参见 


图 8-1 和 图 8-2)。 
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得 益 于 模板 的 使 用 


当 你 第 一 次 开发 动态 Web 站 点 时 ， 创 建 和 使 用 模板 会 非常 困难 。 使 用 模板 的 关键 在 于 要 创建 


一 个 原型 , 并 将 其 切 分 成 不 同 的 部 分 。 然 后 使 用 8.2 市 中 介绍 的 PHP 函 数 ， 


这 些 重 复 的 部 分 就 可 以 
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很 方便 地 包含 到 每 个 页 面 中 ， 然 后 逐 页 面 地 生成 新 内 容 。 下 面 创建 本 章 示例 中 要 使 用 的 模板 , 首 
先 建立 一 个 原型 。 这 个 示例 的 布局 (参见 图 8-3) 是 由 Six Shooter Media (www.sixshootermedia.com) 
的 James Koster 设 计 的 ， 并 且 已 获得 了 使 用 许可 。 


名 lx 











Raise High the Roof Beam! Ne miro net 


Favorite Quotes 
Welcome to a J.D. Salinger Fan Club 





Another Header 




















图 8-3 ”本 章 示 例 的 页 面 设 计 ， 一 个 单独 的 、 静 态 的 HTML 页 面 。 


> 创建 布局 模型 

















(GD 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 HTML 文 档 , 命 名 为 Lemplate.html (参看 脚本 8-1): 


<!DOCTYPE html PUBLIC "“-//W3C//DTD XHTML 1.1//EN" "http://www. 
>W3 .Org/TR/xhtml11/DTD/xhtml11.dtd"> 
<html xmlns="http://www.w3.0o0rg/1999/xhtml" xml:lang="en"> 
<head> 
<title>Raise High the Roof Beam! A J.D. Salinger Fan Club</title> 
<meta http-equiv="content-type" content="text/html; 
"Charset=utf-8" /> 














脚本 8-1 该 脚本 展现 了 网 站 每 个 页 面 的 基本 外 观 


J <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" "http://www.w3.0org/TR/xhtml11/ 

DTD/xhtml1l1.dtd"> 

<html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en"> 

<head> 
<title>Raise High the Roof Beam! A J.D. Salinger Fan Club</title> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<link rel="stylesheet" href="css/1.css" type="text/css" 
media="screen,projection" /> 

7 </head> 

8 <body> 

9 <div id="wrapper"> 





OU 心 wN 
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10 

i <div id="header"> 

2 <p class="description">A J.D. Salinger Fan Club</p> 

18 <hl><a href="index.php">Raise High the Roof Beam!</a></h1l> 

14 <ul id="nav"> 

5 <li><a href="books.php"> Books</a></1i> 

16 <li><a href="#">Stories</a></1i> 

57 <li><a href="#">Quotes</a></1i> 

18 <li><a href="login.php"> Login</a></1i> 

19 <li><a href="register.php"> Register</a></1i> 

20 </ul> 

24 </div><!-- 页 头 --> 

22 

23 <div id="sidebar"> 

24 <h2>Favorite Quotes</h2> 

25 <p class="news">I don't exactly know what I mean by that, but 
I mean it.<br/>- <em>The Catcher in the Rye</em></p> 

26 <p class="news">I privately Say to you, old friend... please 
accept from me this unpretentious bouquet of early-blooming 
parentheses: (((()))).<br/>- <em>Raise High the Roof Beam, 


Carpenters and Seymour: An Introduction</em></p> 
27 </div><!-- 边栏 --> 











28 

29 <div id="content"> 

30 <!-- 可 变 内 容 开 始 。--> 

已 下 <h2>Welcome to a J.D. Salinger Fan Club</h2> 

32 <p>Lorem ipsum dolor sit amet, 
consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et 
dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco 
laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in 
reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. 
Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia 
deserunt mollit anim id est laborum.</p> 

33 <h2>Another Header</h2> 

34 <p>Lorem ipsum dolor sit amet, 
consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et 
dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco 
laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in 
reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. 
Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia 
deserunt mollit anim id est laborum.</p> 

35 <!-- 可 变 内 容 结束 。--> 

36 </div><!-- 内 容 --> 

37 

38 <div id="footer"> 

39 <p>Template design by <a href="http://www.sixshootermedia.com"> 
Six Shooter Media</a>.</p> 

40 <D>&copy; 2011</p> 

光 寺 </div><!-- 页 脚 --> 

42 

43 </div><!-- 完成 页 面 --> 

44 </body> 


45 </html> 
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开发 任何 模板 系统 的 第 一 步 都 是 创建 一 个 模型 文档 一 一 一 个 展示 了 基本 的 页 面 样子 的 实例 。 
一 旦 创建 好 模型 文档 ， 就 可 以 将 其 划分 成 块 了 。 
(2) 添加 CSS 代 码 : 


<link rel="stylesheet" href="css/1.css" type="text/css" 
"media="screen,projection" /> 


对 于 这 个 应 用 程序 , 将 使 用 CSS 进 行 所 有 的 格式 化 和 布局 管理 。CSS 存 储 在 一 个 外 部 文件 中 ， 
通过 1ink 标 签 戏 入 到 页 面 中 。 该 文件 被 简单 命名 为 1.css， 存 储 在 css 文 件 夹 下 。 

注意 ， 你 需要 从 本 书 的 网 站 (wwwLarryUllman.com) 上 下 载 这 个 CSS 文 件 。 这 个 文件 是 本 
书 可 下 载 代码 的 一 部 分 。 

(3) 关闭 HTML head， 打 开 boqy， 创 建 一 个 div 标签 1dq 为 wrapper: 


</head> 
<body> 
<div id="wrapper"> 


如 今 ， 很 多 设计 将 页 面 的 全 部 内 容 放 在 主 div 标 签 之 中 ， 这 样 可 以 很 方便 地 在 浏览 器 窗口 中 
格式 化 所 有 内 容 。 
(4) 创建 页 面 的 header: 


<div id="header"> 
<p class="description">A J.D.Salinger Fan Club</p> 
<hl><a href="index.php">Raise High the Roof Beam!</a></h1l> 
<ul id="nav"> 
<li><a href="books.php"> Books</a></1i> 
<1i><a href="#">Stories</a></1i> 
<li><a href="#">Quotes</a></1i> 
<l1i><a href="login.php"> Login</a></1i> 
<l1i><a href="register.php"> Register</a></1i> 
</ul> 

</div><!-- 页 头 --> 


header 区 域 (CSS 代 码 中 也 已 定义 ) 创建 标语 和 主导 航 链接 ， 这 些 链接 可 以 导航 到 这 个 Web 
应 用 程序 的 其 他 页 面 。 这 些 链 接 引 用 了 4 个 PHP 脚 本 ， 本 章 会 教 你 编写 这 些 脚 本 。 
(5) 创建 页 面 的 边栏 ; 


<div id="sidebar"> 





























<h2>Favorite Quotes</h2> 

<p class="news">I don't exactly know what I mean by that, but I mean it. 
"<br/>- <em>The Catcher in the Rye</em></p> 

<p class="news">I privately say to you, old friend... 
"please accept from me this unpretentious bouquet of early-blooming parentheses: 
»(((()))).<br/>- <em>Raise High the Roof Beam, Carpenters and Seymour: 
An Introduction</em></p> 

</div><!-- 边栏 --> 


在 这 个 最 初 的 模板 中 ,边栏 可 以 用 于 发 布 最 新 消息 、 链 接 次 要 网 页 、 放 置 搜索 框 ， 等 等 。 就 
示例 网 站 而 言 ， 边 栏 用 于 突出 一 些 引 文 。 
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(6) 首先， 标记 特定 于 页 面 内 容 的 开始 : 


<div id="content"> 


<!-- 可 变 内 容 开始 。--> 


到 这 条 注释 为 止 的 所 有 内 容 在 Web 应 用 程序 中 的 每 个 页 面 都 保持 不 变 。 为 了 明确 特定 页 面 内 


容 的 起 始点 (主要 是 为 自 





己方 便 ) ， 这 里 添加 了 一 个 HTML 和 注释。 实际 上 ， 你 可 以 在 这 个 模板 中 


看 到 很 多 注释 ， 指 出 了 这 些 HIML 对 应 着 页 面 中 的 哪 一 部 分 。 但 也 不 要 在 HIML 或 PHP 脚 本 中 添 


加 过 多 的 注释 信息 。 
在 这 之 前 是 content 


区 域 的 开始 。 该 区 域 在 CSS 代 码 中 进行 了 定义 ， 并 恰当 地 定义 了 页 面 主 





内 容 部 分 的 格式 。 换 句 话 说， 每 个 页 面 中 的 内 容 都 将 放置 在 一 个 id 为 content 的 qiv 中 。 


0) 创建 页 面 的 内 容 : 


<h2>Welcome to a J. 


D. Salinger Fan Club</h2> 


<p>Lorem ipsum dolor sit amet...</p> 


对 于 这 个 原型 来 说 , 内 容 只 是 一 系列 标题 和 文本 (实际 脚本 中 的 内 容 要 比 我 在 此 添加 的 东西 


多 得 多 ) 。 


(8) 标记 可 改变 内 容 的 结束 : 
<!-- 可 变 内容 结 束 。--> 


</div><!-- 内 容 --> 


第 7 步 代 码 显 示 的 文本 是 唯一 会 根据 页 面 的 不 同 而 产生 改变 的 文本 。 前 面 的 一 个 HTML 注 释 
引出 了 这 个 区 域 的 起 始 位 置 ， 这 个 注释 指出 了 其 结束 位 置 。 


(9) 添加 footer: 


<div id="footer"> 





<p>Template design by <a href="http://www.sixshootermedia. 
com">Six Shooter Media</a>.</p> 
<p>&copy; 2011</p> 


</div><!-- 页 脚 --> 


footer 包 括 一 些 版 权 声 明 信 息 。 


(10) 完成 HTML 页 面 : 


</div><!-- 完成 页 面 --> 


</body> 
</html> 


注意 : HTML 注 释 标 明了 关闭 的 是 哪个 div 标 签 ， 这 有 助 于 修改 和 维护 模板 。 
(11) 将 文件 保存 为 Lemplate .ntml， 并 在 Web 浏 览 器 中 测试 (参见 图 8-3)。 
以 自己 喜欢 的 方式 完成 了 一 个 原型 之 后 ， 就 可 以 将 其 划分 成 多 个 部 分 ， 用 以 生成 模板 系统 。 


党 创建 页 头 文件 








(1) 如 果 尚 未 打开 template.html (参看 脚本 8-1) ， 在 文本 编辑 器 或 IDE 中 打开 它 。 
(2) 选中 从 HTML 代 码 开头 到 HTML 注 释 <!-- 可 变 内 容 开 始 --> 之 间 的 所 有 内 容 (参见 图 


8-4)。 
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从 站 中 le template.html 
多 MA |“ 门 LastSaved: 12/13/10 10:21:30 AM = 

Pa [Ta] OO @ File Path v : ~/Downloads/typography_paramount/template.html 上 
©|9] template.html +; (no Symbol selected) 3 [WE 大 


<!DOCTYPE html PUBLIC "-//W3CA /ADTD XHTML 1.177ENM "http: / /www.w3.oreg/TR/xhtml11/DT 
了 <html xmlns="http:/ /Www.w3.org/1999/xhtml" xml:lang="en"> 
vv <head> 
<title>Raise High the Roof Beam! A J.D. Salinger Fan Club</title> 
<meta http-equiv="content -type" content="text/html; charset=utf-8" /> 
<link rel="stylesheet" href="css/1.css" type="text/css" media="screen,projecti 
= </head> 
Vv <body> 
Vv «<div id="wrapper"> 


了 <div id="header"> 
<p class="description">A J.D. Salinger Fan Club</p> 
<h1><a href="index.php">Raise High the Roof Beam!</a></h1l> 
<u1 Id="nav"> 
<1i><a href="books.php">Books</a></1i> 
<li><a href="#">Stories</a></1i> 
<li><a href="#">Quotes</a></li> 
<li>¢a href="login.php">Login</a></1i> 
<li>«a href="register.php">Register</a></li> 


- </ul> 
- </div><!-- header --> 
了 <div id="sidebar"> 


<h2>Favorite Quotes</h2> 
<p class="nemws">I don't exactly know what I mean by that, but I mean i 
<p class="News">I privately say to you, old friend... please accept fr 
上 </div><!-- sidebar --> 


村 <div id="content"> 
<!-- BEGIN CHANGEABLE CONTENT. --> 
<h2>Welcome to a J.D. Salinger Fan Club</h2> 
<p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmo 
<h2>Another Header</h2> 
<p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmo 
<!-- END CHANGEABLE CONTENT. --> 

上 </div><!-- content --> 


了 <div id="footer"> 
<p>Template design by <a href="http://ww.sixshootermedia.com">Six Shooter 
<p>acopy; 2811</p> 








- </div><!-- footer --> 
mw <rdiv>c!-- wrapper --» 
-| </body> 
-| </html> 
OO I 
1|1 HTML $$)Unicode (UTF-8) $$ Unix (LF) $$ L] 2,530 / 387 / 45 








图 8-4 ”使 用 示例 文件 ， 选 中 并 复制 前 面 若干 行 代 码 ， 用 于 创建 页 头 文 件 


用 HTML 注 释 指出 特定 于 页 面 的 内 容 ， 这 样 做 的 优点 之 一 在 于 可 以 简化 切 分 模型 的 工作 。 

(3) 复制 代码 

使 用 Edit 菜 单 或 快捷 键 (Windows 上 是 CtrI+C、Macintosh 上 是 Command+C ) ， 将 选中 的 代码 
复制 到 计算 机 的 临时 内 存 〈 也 就 是 剪贴 板 ) 中 。 

(4) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 空白 文档 ， 命 名 为 header .html。 

(5) 将 复制 的 文本 粘贴 到 文档 中 (参看 脚本 8-2)。 


脚本 8-2 ”这 是 一 个 基本 的 页 头 文件 ， 创 建 了 HTML head 信 息 (包含 CSS) 和 body 的 起 始 标签 


1 <!IDOCTYPE html PUBLIC "“-//W3C//DTD XHTML 1.1//EN" 
"http://www.w3.org/TR/xhtml11/DTD/xhtml11 .dtd"> 
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<html Xmlns="http://www.w3.org/1999/xhtm1" xml:lang="en"> 


<head> 
4 <title>Raise High the Roof Beam! 
A J.D. Salinger Fan Club</title> 
5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <link rel="stylesheet" href="css/1.css" type="text/css" 


media="screen,projection" /> 
7 </head> 
8 <body> 
9 <div id="wrapper"> 








10 

11 <div id="header"> 

12 <p class="description">A J.D. Salinger Fan Club</p> 

3 <hl><a href="index.php">Raise High the Roof Beam!</a></h1l> 

14 <ul id="nav"> 

5 <li><a href="books.php">Books 
</a></1i> 

16 <l1i><a href="#">Stories</a></1i> 

17 <li><a href="#">Quotes</a></1i> 

18 <li><a href="login.php">Login 
</a></1i> 

19 <li><a href="register.php"> 
Register</a></1i> 

20 </ul> 

他 二 </div><!-- 页 头 --> 

22 

23 <div id="sidebar"> 

24 <h2>Favorite Quotes</h2> 

25 <p class="news">I don't exactly know what I mean by that, but I mean it. 
<br/>- <em>The Catcher in the Rye</em></p> 

26 <p class="news">I privately say to you, old friend... please accept from me 
this unpretentious bouquet of early-blooming parentheses: (((()))).<br/>- 
<em>Raise High the Roof Beam, Carpenters and Seymour: An Introduction 
</em></p> 

27 </div><!-- 边栏 --> 

28 

29 <div id="content"> 

30 <!-- 可 变 内 容 开 始 。--> 

31 <!-- 脚本 8-2 - header.html --> 

使 用 Edit 菜 单 或 快捷 键 (Windows 上 是 Ctrl+V，Macintosh 上 是 Command+V ) ， 将 所 有 选中 的 

代码 复制 到 新 文档 中 。 


(6) 将 文件 保存 为 header .html。 
既然 页 头 文件 已 经 创建 好 ， 就 可 以 用 同样 的 步骤 创建 页 脚 文件 。 
仿 创 建 页 脚 文 件 


(1) 如 果 尚 未 打开 layout .html (参看 脚本 8-1)， 在 文本 编辑 器 或 IDE 中 打开 它 。 
(2) 选择 从 HTML 广 释 <!-- 可 变 内 容 结束 --> 到 脚本 结束 之 间 的 所 有 内 容 (参见 图 8-5)。 
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人 OA [el template.html [一 
和 T . Last Saved: 12/13/10 10:21:30 AM 
2 及 [Ta QR 总 File Path v : ~/Downloads/typography_paramount/template.html | 上司 
3D template.html +; (no Symbol selected) 3 [ns 大 v 
El 





<div id="content"> 


<!1-- BEGIN CHANGEABLE CONTENT. --> 


<h2>Welcome to a J.D. Salinger Fan Club</h2> 
<p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eius 
<h2>Another Header</h2> 
<p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eius 
<!-- END CHANGEABLE CONTENT. --> 

</div><¢!-- content --> 

<div id="footer"> 
<p>Template design by <a href="http://www.sixshootermedia.com">Six Shoot|| 
<p>acopy; 2811</p> 














上 </div><!-- footer --> 

- <rdiv>c<c!-- wrapper --> | 

= </body> 

-| </html> 人 
OO 5 
3 HTML $$ Unicode (UTF-8) $+$ Unix (LF) $3 2,530 / 387 / 45 





图 8-5 ”继续 使 用 该 示例 文件 ， 选 中 并 复制 最 后 的 若干 行 代码 ， 用 于 创建 页 脚 


(3) 复制 这 部 分 代码 。 
(4) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 空白 文档 ， 命 名 为 footer .html。 
(5) 将 复制 的 文本 粘贴 到 文档 中 (参看 脚本 8-3)。 


脚本 8-3 这 是 一 个 基本 的 页 脚 文件 ， 创 建 了 导航 栏 和 HTML 页 面 的 结束 部 分 











浊 <!-- 脚本 8-3 - footer .htm1l --> 

2 <!-- 可 变 内 容 结束 。--> 

3 </div><!-- 内 容 --> 

4 

5 <div id="footer"> 

6 <p>Template design by <a href="http://www.sixshootermedia.com"> 
Six Shooter Media</a>.</p> 

7 <p>&copy; 2011</p> 

8 </div><!-- 页 脚 --> 

9 

10 </div><!-- 完成 页 面 --> 

11 </body> 

12 </html> 

(6) 将 文件 保存 为 footer .html。 

Vv 提示 


口 在 PHP 中 还 可 以 使 用 更 为 复杂 的 模板 系统 ， 将 设计 与 逻辑 分 开 。Smarty (www.smarty.net) 
可 能 是 最 有 名 的 模板 系统 。 

口 尽管 该 示例 使 用 CSS 进 行 布局 , 但 也 可 以 使 用 表格 来 代 禁 (参见 图 8-6)。 页 头 文件 可 能 需 
要 同时 打开 HTML 页 和 表格 。 每 个 内 容 页 都 会 创建 其 特定 内 容 , 然后 页 脚 文件 需要 同时 完 
成 表格 和 HTML 页面。 要 将 其 转换 为 模板 ， 需 要 将 Page-specific content goes here 之 前 的 代 
码 复制 到 页 头 文件 中 ， 而 将 其 后 的 所 有 代码 复制 到 页 脚 文件 中 。 
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iBGOO0O Table Example 








Top decorative row. 








Header file ends here. 


Navigation Page-specific content goes Blank 
column. here. space. 











Footer file starts here. 














Copyright row. 

















图 8-6 ”这 个 简单 的 示例 展示 了 如 何在 模板 文件 中 使 用 表格 进行 设计 














CSS 模板 


器 ea3g0ggg 人 dogg 人 gogogo 
DO000000000000000000cs000000000000000U000 
DD Webj DUD0O0UUUUCSIUUO 

OUOOODOOOODOODOO0340DODO header[| sidebarl[| content[| footerl[| content 
gg 人 ogo 
加 加 

UUO0O0O0UPHPUCSSUOUODOD PHPU HIMLUOUOUODOD 一 一 PHPUODUOUOODODOUD 
HIMLU CSsUUDOUUOOOOODOUOUOPHPOUOUDOCSSOUODOODOUPHPUDUDUDHIMLUUD 
DUOUOUOOUUOU0O0USSsOUOUOOOUHIMLUUU read 0UU0 








8.2 ”使 用 外 部 文件 


正如 8.1 市 所 述 ， 可 以 将 页 面 划分 成 特定 的 元 素 ， 然 后 使 用 特定 的 函数 将 它们 合并 到 主 PHP 
页 面 中 ,这 样 可 以 市 省 开发 时 间 。 需 要 使 用 的 函数 有 include () 和 require (): 


include ('file.php'); 
require ('file.html'); 


这 两 个 函数 的 工作 方式 相同 ， 它 们 的 区 别 也 不 是 很 大 如 果 include() 函数 失败 了 ，PHP 
脚本 会 生成 一 个 警告 (参见 图 8-7)， 但 继续 运行 。 相 反 ， 如 果 require() 失 败 了 ， 它 会 终止 脚本 
的 执行 (参见 图 8-8)。 

这 两 个 国 数 都 用 来 做 些 什么 呢 ? include () 和 require() 都 会 将 所 引用 的 文件 合并 到 主 文 
件 中 (为 了 清晰 起 见 ， 本 章 将 包含 有 include () 或 require() 代码 行 的 文件 称 作 包 人 多 文件 或 父 
文件 )。 这 两 个 函数 运行 的 结果 将 是 一 样 的， 就 像 被 包含 的 代码 是 父 文件 的 一 部 分 一 样 。 

包含 文件 的 效果 , 就 好 像 被 包含 文件 的 内 容 本 来 就 在 父 脚 本 中 一 样 , 理解 这 一 点 是 用 好 这 个 
函数 的 关键 这 就 意味 着 被 包含 文件 中 没有 放 在 PHP 标 签 里 的 任何 代码 都 将 被 当成 HTML。 而 且 ， 
无 论 被 包含 的 文件 扩展 名 是 什么 结果 都 是 一 样 的 (因为 这 个 文件 只 是 包含 它 的 文件 的 一 个 扩展 )。 
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如 同 回 四 
Waming includeftemplatesfheader html) [function include] failed to 


open stream: Mo such file or directory in /Users/larryullman/Sites 
iphpvqs4/index.php on line 6 


Warming: includeO [funct 
‘templatesiheader.html' for inclusion (include_path="./Applications 如 国 口 | 四 
MANP /biniphp5. 3libiphp") in /Users/larryullmam Sites/phpvqs4 
tindex.php on line 6 


Welcome to a J.D. Salinger Fan Club 


Lorem ipsum dolor st amet, consectetur adipisicing elit, sed do eiusmmod 
tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim 
Yeniam, quis nostrud exercitation ullamco labons nisi ut aliquip ex ea 





include]: Failed opening 





Waming': require(templatesiheader.html) [function reaquire]: failed to open 
streatm: No such file or directory in /Users/larryulhmam Sites/phpvqs4 
iindex.php on line 6 


Fatal error. requireO [function require]: Failed opening required 
‘templates/header.html' (include_path="./Applications/NLANP/biniphp5.3 
































commodo consequat. Duis aute irure dolor in reprehendertt in voluptate tibiphp") in /Users/larryullhmam Sitesiphpvqs4/index.php on line 6 
图 8-7 当 incluge() 失败 时 ， 会 产生 警告 ， 图 8-8” 当 require() 函数 调用 失败 时 , 同时 
但 脚本 能 够 继续 执行 产生 警告 和 错误 ， 并 且 脚 本 停止 执行 





使 用 被 包含 文件 有 很 多 原因 。 可 以 将 自己 定义 的 函数 放 在 一 个 通用 文件 中 (有关 编写 自 有 函 
数 的 更 多 信息 , 请 参见 第 10 章 )。 还 可 以 将 数据 库 访问 信息 放置 在 一 个 配置 文件 中 (参见 第 12 章 )。 
不 过 ， 首 先 我 们 还 是 包含 8.1 市 创建 的 模板 文件 ， 以 便 每 个 页 面 都 能 具备 一 致 的 设计 风格 。 


信使 用 外 部 文件 























(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 文档 ， 命 名 为 index.php。 
(2) 在 文档 起 始 处 输入 PHP 初 始 标签 ， 并 添加 一 些 注释 〈 参 看 脚本 8-4) : 


<?php // 脚本 8-4 - index.php 
/* 这 是 网 站 的 主页 。 
使 用 模板 创建 页 面 布 局 。*/ 





脚本 8-4 ”创建 好 两 个 被 包含 文件 之 后 ， 可 以 使 用 incluae () 函数 将 其 合并 到 父 文件 中 , 以 在 运 


上 上 von 必 wN 
Lm: 


12 
13 


行 时 创建 整个 页 面 


<?php // 脚本 8-4 - index.php 
/* 这 是 网 站 的 主页 。 
使 用 模板 创建 页 面 布局 。*/ 


// 包含 页 头 文件 : 
include('templates/header.html'); 
// 关闭 PHP 节 创建 HIML 内 容 : 


?> 


<h2>Welcome to a J.D. Salinger Fan Club</h2> 

<p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor 
incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud 
exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute 
irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla 
pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia 
deserunt mollit anim id est laborum.</p> 

<h2>Another Header</h2> 

<p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor 
incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud 
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exercitation ullamco laboris nisi ut alidquip ex ea commodo consequat. Duis aute 
irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla 
pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia 
deserunt mollit anim id est laborum.</p> 

14 

15 <?php // 创建 另 一 个 PHP 节 。 

16 include('templates/footer.html'); // Include the footer. 

i 加 实 


注意 ， 有 了 模板 系统 后 ， 脚 本 的 第 一 行 就 是 PHP 标 签 。 这 里 无 需 再 以 HTML 代 码 开始 ， 因 为 
它们 都 被 放 到 了 header .html 文 件 中 。 

(3) 如 果 需 要 的 话 ， 添 加 处 理 错误 消息 的 管理 。 

这 个 主题 已 经 在 第 3 章 中 讨论 过 ， 在 脚本 中 ， 可 以 处 理 错误 也 可 以 不 处 理 错误 。 更 多 信息 请 
参见 第 3 章 ， 这 将 是 在 本 章 中 最 后 一 次 着 重 提 到 这 一 点 。 

(4) 包含 页 头 文件 : 

include('templates/header.html'); 

要 使 用 模板 系统 , 需要 在 这 里 通过 调用 include () 函数 来 包含 页 头 文件 。 由 于 页 头 文件 中 只 
包含 HIML， 因 此 其 中 所 有 的 内 容 会 直接 发 送 到 Web 浏 览 器 ， 就 好 像 这 些 内 容 就 是 当前 文件 的 一 
部 分 。 这 一 行使 用 相对 路 径 来 引用 包含 文件 〈 参 见 框 广 “文件 导航 和 站 点 结构 ”) ， 并 假设 页 头 文 
件 存 储 在 templates 目 录 中 。 

(5) 关闭 PHP 市 并 创建 特定 页 面 的 内 容 : 





























?> 
<h2>Welcome to a J.D. Salinger Fan Club</h2> 
<p>Lorem ipsum dolor sit amet...</p> 


由 于 该 页 面 主要 是 标准 的 HTML, 因此 这 里 暂时 关闭 PHP 市 并 输入 HTML (而 不 是 使 用 print 
将 HTML 发 送 到 Web 浏 览 器 )。 同 时 ， 这 里 包含 的 文字 相对 于 脚本 本 身 来 说 没什么 特殊 意义 。 
(6) 创建 男 一 个 PHP 市 并 请 求 页 脚 文件 : 











<?php 

include('templates/footer.html'); 

为 了 完成 该 页 面 ， 还 需要 包含 页 脚 文 件 (用 于 显示 导航 栏 并 关闭 HTML 代 码 )。 要 完成 这 一 
任务 ， 需 要 创建 一 个 新 的 PHP 节 一 一 在 一 个 脚本 中 可 以 有 多 个 PHP 节 一 一 并 再 次 调用 include () 


(7) 将 文件 保存 为 index .php。 

(8) 在 启用 了 PHP 的 计算 机 或 服务 器 上 的 Web 文 档 目 录 中 ， 创 建 一 个 名 为 templates 的 子 目 录 。 
进一步 分 离 设 计 元 素 和 主 内 容 ， 将 页 头 和 页 脚 文件 放 到 它们 自己 的 目录 中 。 

(9) 将 header .htm1 和 footer .html 放 在 刚才 创建 的 templLlates 目 录 中 。 

(10) 将 index.php 放 置 在 和 templates 文 件 夹 同 级 的 目录 中 。 

(11) 在 启用 了 PHP 的 计算 机 或 Web 服 务 器 的 主 Web 文 档 目 录 中 ， 创 建 一 个 名 为 css 的 子 目 录 。 
这 个 文件 夹 用 于 保存 CSS 脚 本 。 
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(12) 从 本 书 的 随时 代码 中 找到 1 .css 脚 本 ,保存 在 css 文 件 夹 下 (参见 图 8-9)。 
虽然 在 头 文件 中 包括 CSS 脚 本 , 但 对 这 个 脚本 的 引用 必须 相对 于 index.php。 和 毕竟 ， 这 个 页 
面 会 包含 header .html。 











Web 根 目录 盖 ， 
PE | &3 index.php 
| By 二 
> 1 一 
I 全 headerhtml 
r-|[ -| 
1 = 1 
1 templates  【_ 3 footerhtml 
1 
1 
1 
1 SS=3 :> 
el i i i 
| 人 1.css 
攻 
css 











图 8-9 在 启用 了 PHP 的 服务 器 上 ，4 个 文件 和 2 个 文件 夹 的 组 织 结 构图 


索引 页 和 两 个 HTML 页 面 在 计算 机 上 的 相对 位 置 必须 正确 ， 这 样 代码 才能 正常 工作 。 
(13) 在 Web 训 览 器 中 运行 indqex.php (参见 图 8-10) 。 














nitoh Whe Pet be 司 口 四 





Raise High the Roof Beam! wa mow net 


Welcome to a J.D. Salinger Fan Club 





Another Header 








图 8-10 ”该 页 面 是 使 用 被 包含 文件 动态 生成 的 

结果 页 看 上 去 应 该 和 原始 布局 完全 一 致 (参见 图 8-3)。 

(14) 在 Web 训 览 器 中 查看 页 面 源 代码 。 

除了 为 脚本 名 字 添 加 的 注释 和 数字 ， 源 代码 应 该 与 emplate.html 脚 本 (参看 脚本 8-1) 中 
的 源 代码 一 致 。 

ww 提示 

口 模板 系统 中 所 有 的 3 个 文件 (header.html、footer.html 和 index.php) 必须 使 用 相 
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同 的 编码 ， 以 避免 出 现 问题 (关于 编码 的 更 多 信息 ， 请 参见 第 1 章 )。 每 个 文件 的 编码 还 
必须 和 HTML 代 码 中 建立 的 编码 相 匹配 。 
口 在 使 用 require() 和 include() 国 数 时 ， 可 以 使 用 圆 括 号 ， 也 可 以 不 使 用 圆 括号 : 
require 'filename.html'; 
有 了 时 也 可 以 使 用 变量 来 存放 要 包含 的 文件 名 字 : 


require $filename; 








口 





口 


include() 和 require() 都 具有 变种 : include_once() 和 require_once()。 这 两 个 

半数 与 前 两 者 一 一 对 应 ， 但 它们 只 允许 同一 个 文件 被 包含 一 次 (在 同一 个 父 文件 中 ) 通 

第 应 该 尽量 避免 使 用 它们 ， 因 为 这 两 个 函数 会 影响 脚本 的 性 能 。 

如 果 PHP 段 只 包含 一 行 语句 ， 一 般 将 这 条 语句 与 PHP 标 签 写 在 同一 行 : 

<?php include 'filename.html'; ?> 

口 如 果 看 到 了 类 似 于 图 8-7 和 图 8-8 中 的 错误 消息 ， 则 表示 父 脚本 无 法 定位 被 包含 文件 。 这 个 
问题 很 可 能 是 由 于 被 包含 文件 名 拼写 错误 或 路 径 错 误 造 成 的 (例如 ， 使 用 header .html 
赫 换 了 templates/header.html )。 

口 如 果 呈 现 的 网 站 没有 显示 CSS 样 式 ,，HTML 页 面 无 法 找到 相关 文件 。 请 确认 文件 是 否 放 在 

正确 的 文件 夹 下 ， 文 件 名 是 否 正确 ， 引 用 是 否 使 用 的 是 相对 于 index.php 的 链接 。 

文件 的 扩展 名 对 于 被 包含 文件 来 说 并 不 是 很 重要 ， 因 为 它们 不 会 直接 执行 。 一 般 的 经 验 

法 则 是 ， 可 以 安全 地 为 只 包含 或 主要 包含 HTML 的 被 包含 文件 使 用 .html 扩 展 名 (该 扩展 

名 指出 这 是 一 个 HTML 相 关 的 文件 )， 以 及 为 只 包含 或 主要 包含 PHP 的 文件 使 用 .php 扩 展 

名 。 一 些 开发 者 使 用 .inc 扩 展 名 (表示 include), 但 这 样 做 会 带 来 相关 的 安全 风险 。 因 此， 

请 为 任何 包含 有 敏感 信息 (如 数据 库 访 问 参 数 ) 的 文件 使 用 .php 扩 展 名 。 当 然 ， 对 于 需 

要 直接 执行 的 PHP 脚 本 ， 也 要 使 用 .php 扩 展 名 。 

口 外 部 文件 的 另 一 种 比较 好 的 使 用 方式 是 将 错误 设置 代码 放 在 其 中 ， 这 样 设置 的 变更 就 能 

应 用 于 Web 站 点 中 的 每 一 个 页 面 了 。 


口 


























口 

















文件 导航 和 站 点 结构 
DO 
UwebuUOOOUO0OU0000000000000000000000000000050 
国 


include('C:\inetpub\wwwfiles\file.php'); 
include('/Users/larry/Sites/file.php'); 


加 加 上 轩辕 
加 
加 


include('file.php'); 
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iale ool (Lo VE) 


D0000000000000000000000000000000000000 
8-9[ 

include('templates/header.html') 

国生 

nealudelt mv/tiTe nh 

国人 
国 gogogdgggoggogg 人 gggooggdggooggogggogogoog 
加 
加 加 四 icmuas IEessacs 上 加 加 本 眼 本 下 四 四 本 加 加 加 加 加 :加 国 加 国 国 国 国 加 回国 国 
DOOg0g00go000o000g0oo 


8.3 使 用 常量 


本 书 已 经 讨论 了 PHP 中 的 很 多 数据 类 型 : 基本 数值 、 字 符 串 和 数组 。 常 量 是 另外 一 种 数据 类 
型 ， 但 和 变量 不 同 ， 它 们 在 脚本 的 执行 过 程 中 会 一 直 保持 其 初始 值 。 一 旦 设 定 了 常量 值 ， 就 不 能 
再 更 改 它 了 。 

只 能 通过 为 其 赋值 来 创建 常量 。 和 变量 使 用 赋值 运算 符 〈=) 进行 赋值 不 同 ， 常 量 的 赋值 需 
要 使 用 aefine () 函数 : 

define('CONSTANT_ NAME', value); 

请 注意 , 一 条 经 验 法 则 是 ,全 部 使 用 大 写字 母 来 命名 常量 ， 尽管 并 不 是 必须 这 样 做 。 更 重要 
的 是 ， 常 量 不 像 变量 那样 在 字 首 使 用 美元 符号 (因为 常量 不 是 变量 )。 这 里 有 两 个 常量 : 


define ('PI', 3.14); 
define ('CURRENCY', 'euros'); 


对 于 value 来 说 ， 字 符 串 需 要 用 引号 括 起 来 ， 而 数值 不 需要 。 
第 量 的 引用 通常 很 简单 ; 

print CURRENCY 

number_format (PI, 1); 


但 在 引号 内 部 使 用 常量 时 有 些 复杂 ， 这 里 无 法 打印 出 单 引 号 或 双 引 号 内 部 的 常量 值 ， 如 ( 参 
图 8-11 ) : 


print "The cost is 468 CURRENCY"; 
print 'The cost is 468 CURRENCY'; 





























污 

















The cost is 468 CURRENCY 














图 8-11 无 法 打印 出 单 引 号 或 双 引 号 内 部 的 常量 值 
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这 里 需要 使 用 连接 或 使 用 多 个 print 语 句 : 


print 'The cost is 468 ' .CURRENCY; 


渤 


print 'The cost is 468 ' 
print CURRENCY 


请 不 要 混淆 ， 和 用 于 创建 常量 的 define() 函数 一 同 使 用 的 还 有 defined() 国 数 ， 如 果 提 交 
的 常量 已 定义 ， 则 这 个 函数 会 返回 TRUE。defined() 函数 常常 用 在 条 件 语句 中 : 

if (defined('CONSTANT NAME')) { ... 

作为 使 用 常量 的 一 个 示例 , 我 们 将 为 前 面 的 示例 程序 添加 一 种 能 够 为 每 个 页 面 显 示 不 同 标题 
(将 出 现在 浏览 器 窗口 的 顶部 ) 的 功能 。 为 了 完成 这 一 功能 ， 需 要 在 父 脚本 中 定义 一 个 常量 ， 然 
后 在 页 头 文件 中 将 父 脚 本 打印 出 来 。 这 种 技术 是 可 行 的 ， 只 要 在 incluqde() 语 句 或 require() 
语句 中 恰当 地 引用 了 这 些 文件 ， 结 构 化 的 应 用 程序 就 能 运行 得 很 好 ， 并 带 来 易于 维护 等 优点 。 

信使 用 常量 的 步骤 

(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 PHP 文 要 ， 命 名 为 books .php (参看 脚本 8-5): 


<?php // 脚本 8-5 - books .php 

















脚本 8-5 ”这 个 脚本 不 但 使 用 和 ingdex.php (参看 脚本 8-4) 相同 的 模板 系统 ， 而 且 还 使 用 了 一 
个 用 于 标识 页 面 标 题 的 常量 

<?php // 脚本 8-5 - books .php 

/* 这 个 页 面 显 示 J.D. Salinger 的 著作 。*/ 


// 设置 页 面 标题 并 包含 页 头 文 件 : 
define ('TITLE', 'Books by J.D. Salinger'); 
include('templates/header.html'); 


// 关闭 PHP 节 创建 HTML 内 容 : 


和 


Ooo 和 ORODP 


<h2>J.D. Salinger's Books</h2> 
> 

<li>The Catcher in the Rye</1i> 

<li>Nine Stories</1i> 

<li>Franny and Zooey</1i> 

<li>Raise High the Roof Beam, Carpenters and Seymour: An Introduction</1i> 
</ul> 


oo vam 必 wb 口 





Ko) 


1 <?php include('templates/footer.html'); ?> 
| 2 时 

(2) 将 页 面 标题 定义 为 常量 : 

define ('TITLE', 'Books by J.D. Salinger'); 


这 里 定义 了 一 个 常量 ， 名 为 TITLE， 并 为 其 赋值 Books by J.D. Salinger。 
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(3) 包含 页 头 文 件 : 
include('templates/header.html'); 


该 脚本 使 用 和 所 有 其 他 脚本 相同 的 页 头 文件 , 但 稍 后 将 对 其 进行 少许 修改 , 将 常量 引入 进来 。 
(4) 关闭 PHP 节 ， 创 建 HTML: 


? 
<h2>J.D. Salinger's Books</h2> 
<ul> 
<li>The Catcher in the Rye</1i> 
<li>Nine Stories</1i> 
<li>Franny and Zooey</1i> 
<li>Raise High the Roof Beam, 
Carpenters and Seymour: An 
Introduction</1i> 
</ul> 


这 里 给 出 的 内 容 非 常 简单 ， 但 正好 达到 适用 于 页 面 的 目的 。 
(5) 创建 一 个 新 的 PHP 市 并 包含 页 脚 文件 : 
<?php include('templates/footer.html'); ?> 


在 8.2 节 的 提示 中 说 过 ,由 于 PHP 代 码 只 有 一 条 语句 ， 可 以 将 这 条 语句 与 PHP 开 始 和 关闭 标签 








写 在 同一 行 。 但 要 注意 在 要 执行 的 代码 (include () ) 和 标签 之 间 要 加 一 个 空格 。 


(6) 将 文件 保存 为 books .php。 
为 了 使 用 常量 ， 接 下 来 还 需要 修改 header .html 文 件 。 


这 打印 一 个 常量 








(1) 在 文本 编辑 器 或 IDE 中 打开 header .html (参看 脚本 8-2)。 
(2) 删除 出 现在 title 标 签 中 的 Raise High the Roof Beam! A J. D. Salinger Fan Club (参见 第 4 





既然 页 面 标题 将 根据 页 面 的 不 同 有 所 变化 ， 就 不 需要 将 其 硬 编码 到 页 面 中 了 。 
(3) 在 删除 文本 的 地 方 (title 标 签 中 )， 添加 下 面 的 脚本 (参看 脚本 8-6): 


<?php 
if (defined('TITLE')) { 

print TITLE; 
} else { 

print 'Raise High the Roof Beam! A J.D. Salinger Fan Club'; 
} 


?> 











脚本 8-6 header .html 文 件 经 过 了 修改 ,这 样 它 就 可 以 根据 常量 是 否 存在 以 及 常量 的 值 来 设置 


页 面 标 题 


1 <!IDOCTYPE html PUBLIC "“-//W3C//DTD XHTML 1.1//EN" 
"http://www.w3.0org/TR/xhtml11/DTD/xhtml11 .dtd"> 

2 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en"> 

3 <head> 
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4 <title><?php // 打印 页 面 标题 。 

5 if (defined('TITLE')) { // 检查 标题 是 否 已 定义 。 

6 print TITLE; 

7 } else { // 标题 没有 定义 。 

8 print 'Raise High the Roof Beam! A J.D. Salinger Fan Club'; 

9 } 

10 ?></title> 

11 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
12 <link rel="stylesheet" href="css/1.css" type="text/css" 


media="screen,projection" /> 
3 </head> 
14 <body> 
15 <div id="wrapper"> 
6 
7 
8 


<div id="header"> 
1 <p class="description">A J.D. Salinger Fan Club</p> 
19 <hl><a href="index.php">Raise High the Roof Beam!</a></h1l> 





20 <ul id="nav"> 

21 <li><a href="books.php">Books</a></1i> 

22 <l1i><a href="#">Stories</a></1i> 

23 <li><a href="#">Quotes</a></1i> 

24 <li><a href="login.php">Login</a></1i> 

25 <li><a href="register.php">Register</a></1i> 

26 </ul> 

27 </div><!-- 页 头 --> 

28 

29 <div id="sidebar"> 

3@ <h2>Favorite Quotes</h2> 

沪 和 <p class="news">I don't exactly know what I mean by that, but I mean it. 
<br/>- <em>The Catcher in the Rye</em></p> 

2 <p class="news">I privately say to you, old friend... please 
accept from me this unpretentious bouquet of early-blooming 
parentheses: (((()))).<br/>- <em>Raise High the Roof Beam, 





Carpenters and Seymour: An Introduction</em></p> 
33 </div><!-- 边栏 -- 


34 

35 <div id="content"> 

36 <!-- 可 变 内 容 开 始 。--> 

37 <!-- 脚本 8-6 - header.html --> 


为 了 能 i 上 PHP 创 建 页 面 标 题 ， 首 先 需 要 在 title 标 签 中 打开 一 个 PHP 代 码 节 。 然 后 使 用 一 个 
条 件 语句 检查 是 否 已 经 定义 了 TITLE 常 量 。 如 果 已 经 定义 ， 则 打印 它 的 值 并 作为 页 面 标 题 。 如 果 
没有 定义 TITLE， 则 打印 一 个 默认 标题 。 

(4) 将 文件 保存 为 header .html。 

(5) 将 books .php 和 header .html 上 传 到 启用 了 PHP 的 服务 器 上 。books .php 这 个 新 脚本 应 
该 和 index .php 位 于 同一 目录 ， 而 header .htm1 应 该 替换 掉 之 前 的 版 本 ， 并 和 footer .html 一 
样 都 位 于 目录 templates 中 。 

(6) 在 Web 浏 览 器 中 运行 books .php (参见 图 8-12)。 

(7) 在 Web 浏 览 器 中 查看 index.php (主页 ) (参见 图 8-13)。 
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外 
Raise High the Roof Beaml 
贸 
JD. Salingers Books Raise High the Roof Beaml 
RD ea Welcome to a J.D. Salinger Fan Club 
图 8-12 ”books 页 面 使 用 PHP 常 量 来 创建 标题 图 8-13 ”由 于 index 页 中 没有 定义 了 ITI ， 所 以 


E 溃 量 

该 页 面 的 标题 使 用 了 默认 值 (这 是 由 于 在 
脚本 8-6 中 使 用 了 条 件 语句 ) 
(8) 如 果 有 兴趣 ， 可 以 在 index .php 中 添加 一 个 常量 定义 ， 修 改 其 标题 。 
Vv 提示 
口 除了 要 省 略 掉 美元 符号 之 外 ， 常 量 的 正式 命名 规则 和 变量 是 完全 一 样 的。 常量 名 称 必须 
以 字母 开头 ， 可 以 包含 字母 、 数 字 和 下 划 线 的 任意 组 合 ， 并 且 是 区 分 大 小 写 的 。 
口 PHP 有 很 多 预定 义 的 常量 。 包 括 PHP_VERSION (正在 运行 的 PHP 的 版 本 ) 和 PHP_0S ( 服 
务 器 的 操作 系统 )。 
口 在 第 9 章 中 ， 会 介绍 另外 一 个 常量 一 一 SID (表示 session 的 ID ) 。 
口 使 用 常量 的 另外 一 个 好 处 是 ， 它 们 是 全 局 作用 域 的 。 在 阅读 完 第 10 章 中 讲述 这 一 部 分 的 
内 容 之 后 ， 读 者 会 对 此 有 更 多 认识 。 
口 不 仅 常 量 的 值 不 能 修改 ， 常 量 本 身 也 不 能 删除 。 另 外 ， 一 个 常量 只 能 包含 一 个 单独 的 值 ， 

这 和 数组 不 同 ， 但 是 和 字符 串 或 数字 是 一 样 的 。 


8.4 使 用 日 期 和 时 间 


PHP 具 备 一 些 用 于 操作 日 期 和 时 间 的 函数 ， 其 中 最 重要 的 就 是 aate () 。date() 图 数 做 的 唯 
一 的 事情 就 是 基于 提供 给 它 的 参数 , 返回 格式 化 的 日 期 和 时 间 信 息 , 但 当 了 解 到 它 有 多 么 有 用 后 ， 
你 一 定 会 感到 惊讶 ! aate () 函数 的 基本 使 用 方法 是 : 


date('formatting'); 


可 以 用 于 进行 格式 化 的 选项 有 很 多 ， 表 8-1 列 出 了 所 有 的 选项 。 这 些 参数 还 可 以 组 合 一 一 例 


如 ，dqate('1 F j，Y') 可 以 返回 Wednesday January 26，2011。 
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表 8-1 Date() 函数 格式 






























































字 符 含义 示 例 
Y 4 位 数字 表示 的 年 份 2011 

区 2 位 数字 表示 的 年 份 11 

是 否 为 国 年 1 (表示 “是 ”) 
沪 1 或 2 位 数字 表示 的 月 份 2 

wm 2 位 数字 表示 的 月 份 02 

F 月 份 February 

M 3 个 字母 表示 的 月 份 Feb 

j 1 或 2 位 数字 表示 的 月 份 中 的 一 天 8 

a 2 位 数字 表示 的 月 份 中 的 一 天 08 

1 (小 写 的 L) 一 周 中 的 某 一 天 Monday 

An 3 个 字母 表示 的 一 周 中 的 某 一 天 Mon 

w 1 位 数字 表示 的 一 周 中 的 某 一 天 0 (星期 日 ) 
“ 一 年 中 的 某 一 天 : 0~365 189 

t 月 份 中 有 多 少 天 31 

8 2 个 字符 表示 的 天 数 英文 序数 词 后 级 rd 

9 小 时 数 ，! 或 2 位 数字 表示 的 12 小 时 制 格式 6 

G 小 时 数 ，1 或 2 位 数字 表示 的 24 小 时 制 格 式 18 

h 小 时 数 ，2 位 数字 表示 的 12 小 时 制 格 式 06 

H 小 时 数 ，2 位 数字 表示 的 24 小 时 制 格 式 18 

i 分 钟 数 45 

S 秒 数 18 

um 毫秒 数 1234 

a am 或 pm am 

A AM 或 PM PM 

y 从 epoch 开 始 的 秒 数 1048623008 
e 时 区 UTC 

I (大 写 的 i) 是 否 为 夏令 时 1 (表示 “是 ”) 
0 与 GMT 之 间 的 时 差 +0600 


date() 国 数 可 以 接受 另 一 个 参数 ， 这 个 参数 称 作 时 间 稚 (timestamp)。 时 间 惟 是 一 个 数字 ， 
表示 从 1970 年 1 月 1 日 午夜 起 计算 的 秒 数 一 一 这 一 时 刻 也 称 作 epoch。time () 函数 可 以 返回 当前 时 
刻 的 时 间 惟 。mktime () 国 数 可 以 返回 一 个 给 定 的 时 间 和 日 期 的 时 间 惟 : 

mktime (hour, minute, second, month, day, year); 
因此 下 面 的 代码 : 


$sts = mktime(12, 30, 0, 11, 5, 2011); 
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可 以 将 从 epoch 到 2011 年 11 月 5 日 12:30 经 过 的 秒 数 赋值 给 Sts 变 量 。 这 个 数值 可 以 传递 给 date () 
函数 ， 如 下 所 示 : 

date('D', S$ts); 

这 会 返回 Sat， 它 是 用 3 个 字母 表示 的 一 周 中 的 某 一 天 。 

对 于 PHP5.1, 需要 在 调用 任何 日 期 或 时 间 相关 的 函数 之 前 , 设置 服务 器 的 时 区 。 要 完成 这 一 
任务 ， 请 使 用 : 


date_default_timezone_set (timezone); 

timezone 的 值 是 类 似 于 America/New_York 或 Pacific/Auckland 这 样 的 字符 串 。 这 些 值 
太 多 了 ， 无 法 在 这 里 列 出 〈 仅 非洲 就 有 超过 50 个 时 区 ) ， 可 以 在 PHP 手 册 中 查 到 所 有 的 值 。 如 果 
没有 完成 这 一 步 ， 则 会 看 到 错误 (参见 图 8-14)。 





Warning: date() [function.datej: It is not safe to 
rely on the System's timezone settings. YOU are 
*required* to use the date.timezone setting or the 
date_default_timezone_set0 function. In case YOU 
used any of those methods and you are stil 
getting this warning, you most likely misspelled 
the timezone identifier. We selected 

‘America/ New_York’ for "EST/-5.0/no DST instead 
in 
/Users/larryullman/Sites/phpvqs4/templates/header.html 
on line 38 











图 8-14 对 于 PHP 5.1， 当 没有 设置 时 区 就 调用 日 期 或 时 间 函 数 时 ， 会 生成 一 个 通知 


为 了 演示 date () 函数 ， 我 们 来 修改 一 下 页 脚 文件 ， 便 能 显示 当前 的 日 斯 和 时 间 (参见 图 
8-15 ) 。 























Favorite Quotes 


1 don't exactfy know what | mean by that. but | 
mean it. 
- The Catcher in the Rye 


lprivately say to you, old friend... Please 
accept from me this unpretentious bouquet of 
early-blooming parentheses: tt). 

- Kaise High the Roorf Beam, Carpenters ancf 
Sepmovur. An ntroovuection 


11:58 am Monday December 13 














| 


8-15 ”使 用 date() 国 数 ，Web 站 点 现在 可 以 在 边栏 中 显示 日 期 和 时 间 
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瀛 使 用 aate() 


(1) 在 文本 编辑 器 或 IDE 中 打开 header .html (参看 脚本 8-6)。 
(2) 在 </div> 关 标签 之 前 ,添加 下 面 的 代码 (参看 脚本 8-7) : 


<p><?php 








脚本 8-7 修改 后 的 header .html 文 件 使 用 date() 函数 来 打印 当前 日 期 和 时 间 


1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" "http://www.w3.org/TR/xhtml11/ 
DTD7Yxhtmll1l..dtd"> 








<html xmlns="http://www.w3.o0org/1999/xhtml" xml:lang="en"> 

3 <head> 

4 <title><?php // 打印 页 面 标题 。 

S if (defined('TITLE')) { // 检查 标题 是 否 已 定义 。 

6 Print. ‘TITDE::; 

} else { // 标题 没有 定义 。 

8 print 'Raise High the Roof Beam! A J.D. Salinger Fan Club'; 
9 } 

10 ?></title> 


<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<link rel="stylesheet" href="css/l1.css" type="text/css" media="screen, 
projection" /> 
3 </head> 
4 <body> 
15 <div id="wrapper"> 
6 
7 
8 


<div id="header"> 
1 <p class="description">A J.D. Salinger Fan Club</p> 
19 <hl><a href="index.php">Raise High the Roof Beam!</a></h1l> 








20 <ul id="nav"> 

2 <li><a href="books.php">Books</a></1i> 

22 <l1i><a href="#">Stories</a></1i> 

23 <li><a href="#">Quotes</a></1i> 

24 <li><a href="login.php">Login</a></1i> 

25 <li><a href="register.php">Register</a></1i> 

26 </ul> 

27 </div><!-- 页 头 --> 

28 

29 <div id="sidebar"> 

30 <h2>Favorite Quotes</h2> 

全 二 <p class="news">I don't exactly know what I mean by that, but I mean it. 
<br/>- <em>The Catcher in the Rye</em></p> 

3 <p class="news">I privately say to you, old friend... please accept from me 
this unpretentious bouquet of early-blooming parentheses: (((()))).<br/>- 
<em>Raise High the Roof Beam, Carpenters and Seymour: An Introduction 
</em></p> 

33 <p><?php // 打印 当前 日 期 和 时 间 …… 

34 // 设置 时 区 : 

35 date default timezone set('America/New York'); 
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37 // 打印 日 期 和 时 间 : 

38 print date('g:i a lrF j'); 
39 PRY BS 

40 </div><!-- 边栏 --> 

41 

42 <div id="content"> 

43 <!-- 可 变 内 容 开 始 。--> 

44 <!-- 脚本 8-7 - header.html --> 





第 一 个 HIML 标 签 对 日 期 和 时 间 稍 微 进行 了 一 下 格式 化 ， 并 将 其 设置 为 斜体 [使 用 “强调 ” 
(em) 标签 ]。 然 后 打开 一 个 PHP 节 ， 这 样 才能 调用 aate () 函数 。 
(3) 建立 时 区 : 
date_default_timezone_set('America/New_York'); 
在 调用 date() 之 前 ， 必 须 设置 时 区 。 要 查找 时 区 ， 请 参见 www.php.net/timezones。 
(4) 使 用 aate () 函数 来 打印 当天 的 日 期 和 时 间 : 
print date('g:i a 1l Fj'); 
使 用 表 8-1 中 列 出 的 格式 化 参数 ， 这 里 的 date () 函数 将 返回 一 个 类 似 4:15 pm Tuesday 
February 22 这 样 的 值 。 这 个 值 将 会 被 立即 打印 出 来 。 
(5) 关闭 PHP 市 ， 完 成 HTML 代 码 : 
2></p> 
(6) 将 文件 保存 为 header .html， 放 置 在 启用 了 PHP 的 服务 器 上 的 templates 目 录 中 ， 并 在 
Web 浏 览 器 中 进行 测试 (参见 图 8-15)。 
Vv 提示 
口 由 于 PHP 是 一 种 服务 器 端的 技术 , 因此 这 些 函 数 反映 的 是 服务 器 上 的 日 期 和 时 间 。 要 获取 
客户 端 ( 换 句 话说 ,就 是 用 于 浏览 页 面 的 Web 浏 览 器 所 在 的 计算 机 ) 的 时 间 ， 则 必须 使 用 
JavaScript。 
口 服务 器 的 时 区 也 可 以 在 PHP 配 置 文件 中 设置 (参见 附录 A)。 在 这 里 建立 时 区 要 优 于 分 别 
在 每 个 脚本 中 做 这 项 工作 。 
口 PHP 5.3 中 加 入 了 一 个 新 的 创建 和 操作 日 期 和 时 间 的 方式 一 一 DateTime 类 。 虽 然 很 有 用 ， 
但 这 个 新 工具 需要 使 用 者 熟悉 面向 对 象 编程 ， 这 超出 了 基础 教程 的 范围 ， 这 里 不 作 过 多 
介绍 。 


8.5 再 谈 使 用 PHP 处 理 HTML 表单 


到 目前 为 止 ， 本 书 中 所 有 的 示例 都 是 用 两 个 分 离 的 脚本 来 处 理 HTML 表 单 : 一 个 用 于 显示 表 
单 ， 男 一 个 用 于 接收 和 处 理 表单 数据 。 这 种 方法 并 没有 什么 不 对 , 但 在 一 个 脚本 中 完成 整个 流程 
则 更 为 有 益 。 为 了 让 一 个 页 面 同时 显示 和 处 理 表 单 ， 可 以 使 用 一 个 条 件 语 句 (参见 图 8-16): 


IE (/* form has been submitted */) { 
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<?php 
include( 'template/header.html' ); 


if (/* form has been submitted */) { 








聆 证 


(} else { | 


} 


include( 'template/footer.html '); 

















?> 





脚本 

















图 8-16 ”这 个 流程 图 说 明 如 何在 一 个 PHP 脚 本 中 显示 和 处 理 HTML 表 单 


有 很 多 种 方式 可 以 检测 表单 是 否 已 经 被 提交 ， 一 种 方法 是 检查 是 否 为 某 些 变量 设置 了 值 : 























if (isset($_ POST['something'])) { . 


然而 ,如果 用 户 在 没有 完成 表单 的 情况 下 就 进行 了 提交 ， 则 可 能 没有 设置 变量 的 值 (取决 于 


对 应 表单 元 素 的 类 型 )。 我 的 解决 方法 是 在 表单 中 加 入 一 个 隐藏 文本 框 来 进行 检测 ， 这 种 方法 更 
可 靠 : 


<input type="hidden" name="submitted" value="true" /> 


该 隐藏 输入 框 的 唯一 目的 就 是 可 靠 地 指出 表单 已 经 被 提交 。 为 了 检测 这 一 点 , 用 于 处 理 表单 
的 PHP 代 码 可 以 使 用 下 面 的 条 件 语 句 : 


if (isset($_POST['submitted'])) { ... 


另 一 种 检查 表单 提交 情况 的 方法 是 0D 口 口 吕 DOD 00 。 当 有 一 个 表单 , 在 提交 后 返回 相同 
的 页 面 时 ， 脚 本 会 由 两 种 不 同类 型 的 请 求 组 成 (参见 图 8-17)。 第 一 个 请 求 是 加 载 表 单 的 GET 请 
求 。 这 是 大 多 数 Web 页 面 都 会 有 的 标准 请 求 。 当 表单 被 提交 时 ， 它 的 action 属 性 指向 同一 个 页 
面 ， 这 时 会 产生 第 二 个 请 求 ， 此 时 是 POST 请 求 (假设 表单 使 用 了 POST 方 法 )。 因 此 ， 可 以 通过 
检查 表单 的 请 求 类 型 来 测试 表单 的 提交 情况 ， 使 用 $_SERVER 数 组 : 


if ($_SERVER['REQUEST METHOD'] == 'POST') { ... 


作为 示例 ， 我 们 在 这 里 将 建立 一 个 基本 的 登录 表单 。 
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<?php 
include('template/header .html ' ) ; 


if (/* form has been submitted */) { 























© 
POST 
GET 请 求 目 i 提交 请 求 
ee > 
@ 


include( 'template/footer.html ' ); 
?> 





脚本 











图 8-17” 当 在 一 个 PHP 脚 本 中 同时 显示 和 处 理 HTML 表 单 时 ， 有 两 种 请 求 脚本 的 方式 


> 使 用 一 个 页 面 来 显示 和 处 理 表单 

















(GD 在 文本 编辑 器 或 IDE 中 打开 一 个 新 的 PHP 文 档 ， 命 名 为 10ogin .php (参看 脚本 8-8): 


<?php // 脚本 8-8 - login.php 








脚本 8-8 ”这 个 登录 页 用 于 两 种 目的 。 它 会 显示 登录 表单 并 处 理 其 提交 的 文档 





1 <?php // 脚本 8-8 - login.php 

2  /* 页 面 可 以 用 于 用 户 登 录 (理论 上 )。*/ 

3 

4 ” // 设置 页 面 标题 并 包含 页 头 文 件 : 

5 define('TITLE', 'Login'); 

6 include('templates/header.html'); 

7 

8 ”// 打印 一 些 介绍 文本 : 

9 print '<h2>Login Form</h2> 

10 <p>Users who are logged in can take advantage of certain features like this, 
that, and the other thing.</p>'; 

11 

12 // 检查 表单 是 否 已 提交 : 

13 if ($_SERVER['REQUEST _ METHOD == 'POST') { 

14 

15 // 处 理 表单 : 

16 if ( (!empty($ POST['email'])) && (!empty($ POST['password'])) ) { 

1 

18 if ( (strtolower($ POST['email']) == 'me@example.com') 

&& ($_ POST['password'] == 'testpass') ) { // 相等 ! 
19 
20 print '‘'<p>You are logged in!<br/>Now you can blah, 


blah, blah...</p>'; 
2 
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22 } else { // 不 相等 ! 
23 
24 print '<p>The submitted email address and password 


do not match those on file!<br/>Go back and try 
again.</p>'; 


25 

26 } 

27 

28 } else { // 表单 未 填 完 整 。 

29 

30 print '<p>Please make sure You enter both an email address 
and a password!<br/>Go back and try again.</p>'; 

3 

32 } 

33 

34 } else { // 显示 表单 。 

35 

36 print '‘'<form action="login.php" method="post"> 

37 <p>Email Address: <input type="text" name="email" size="20" /></p> 

38 <p>Password: <input type="password" name="password" 

size="20" /></p> 

39 <p><input type="submit" name= "submit" value="Log In!" /></p> 

40 </form>'; 

41 

42 } 

43 

44 include('templates/footer.html'); // Need the footer. 

45 ?> 

(2) 将 页 面 标题 定义 为 一 个 常量 ， 并 包含 页 头 文件 : 

define('TITLE', 'Login'); 





include('templates/header.html'); 


可 以 使 用 本 章 前 面 介绍 的 常量 系统 ， 为 页 面 设置 自己 的 唯一 页 面 标题 。 
(3) 添加 一 些 介 绍 性 文本 : 


print '<h2>Login Form</h2> 
<p>Users who are logged in can take advantage of certain 
features like this, that, and the other thing.</p>'; 


这 些 文本 位 于 主 条 件 语 句 之 外 ， 不 论 表 单 是 正在 被 显示 还 是 已 经 被 提交 ， 它 会 一 直 显 示 在 
Web 浏 览 器 中 。 因为 这 个 脚本 的 核心 被 包含 在 一 个 PHP 条 件 语 名 中 , 所 以 需要 用 PHP 打 印 出 HTML 
表单 ， 而 不 是 像 前 两 个 示例 (ingdex.php 和 books .php) 那样 将 其 放 在 PHP 代 码 之 外 。 

(4) 开启 一 个 条 件 语句 ， 检 测 表 单 是 否 已 经 被 提交 : 

if ($_SERVER['REQUEST METHOD'] == 'POST') { 

要 测试 表单 已 经 被 提交 的 情况 , 需要 检查 $_POSTI[' submitted'] 变 量 是 否 等 于 POST (区 
大 小 写 )。 

(5) 创建 一 对 在 套 的 条 件 语句 ， 处 理 表 单数 据 : 


If ( (iempty($_POST['email'])) && (!iempty($_POST['password'])) ) { 


全 让 











3 
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if ( (strtolower($_POST['email']) == 'me@example.com') && 
»($_POST['password'] == 'testpass') ) { 
print '<p>You are logged in!<br/>Now you can blanh, 
blah, blah...</p>'; 
} else { // 不 相等 ! 
print '<p>The submitted email address and password do not 
"match those on file!<br/> Go back and try again.</p>'; 
} 
} else { 
print '<p>Please make sure you enter both an email address 
"and a password!<br/>Go back and try again.</p>'; 


; 


这 些 条 件 语句 用 于 处 理 表 单数 据 。 第 一 个 条 件 语句 检查 了 Email 地 址 和 密码 变量 是 否 具有 值 。 
如 果 没 有 ， 会 显示 一 个 消息 (Please make sure…)。 在 第 一 个 条 件 语句 中 ， 另 一 个 条 件 语句 检查 
了 Email 地 址 是 否 等 于 me@example com 以 及 密码 是 否 等 于 testpass。 如 果 相 等 ， 则 认为 用 户 已 登录 
(现在 介绍 如 何 存储 和 检索 用 户 信息 有 点 超前 )， 否 则 ， 会 显示 一 条 消息 ， 表 示 输入 了 错误 的 值 。 
要 确保 在 这 里 的 条 件 语句 中 使 用 的 是 相等 运算 符 (==) 而 不 是 赋值 运算 符 (=)， 这 是 一 种 常见 
的 错误 ,另外 , 即便 用 户 将 Email 地 址 输入 成 Me @example.com 或 其 他 大 小 写 组 合 方式 也 没有 关系 ， 
因为 这 里 在 检查 相等 性 之 前 ， 首 先 对 Email 地 址 应 用 了 strtolower () 函数 。 
(@) 完成 主 条 件 语句 : 


} else { // 显示 表单 。 
print '<form action="login.php" method="post"> 
<p>Email Address: <input type="text" name="email" size="20" /></p> 
<p>Password: <input type= "password" name="password" size="20" /></p> 
<p><input type="submit" name= "submit" value="Log In!" /></p> 
</form>'; 

















} 
这 段 代 码 结束 了 主 条 件 语 句 ， 即 检查 表单 是 否 已 经 被 提交 的 条 件 语 句 。 如 果 未 被 提交 ， 则 会 
显示 表单 。 这 个 表单 本 身 非 常 简单 (参见 图 8-18)。 








Loegmn Form 


Users who are logged in can take advantage of certain features like this, 


that, and the other thing 
Email Address 


Password 


图 8-18 这 个 简单 的 登录 页 接受 一 个 Email 地 址 和 一 个 密码 














8.5 0U0U000 PHPUU HIMLOD 193 





为 了 避免 混淆 ， 即 使 表单 method 属 性 有 一 个 值 为 post (全 部 小 写 ), 在 检查 表单 提交 情况 时 ， 
请 求 方法 的 值 仍 应 为 POST (全 部 大 写 )。 
(7) 请 求 页 脚 文件 并 完成 PHP 页 面 : 


include('templates/footer.html'); 
?> 


(8) 将 文件 保存 为 1ogin.php， 放 置 在 和 inaex.php 相 同 的 目录 中 ， 然 后 在 Web 浏 览 器 中 进 
行 测试 〈 参 见 图 8-19、 图 8-20 和 图 8-21 ) 。 

















Login Form Login Form 


Users who are logged in can take advantage of certain features like this, 






1 can take advantage of certain features like this, 





that, and the other thing 


You a 09 C 1 
Oy. are:logged.in Please make sure YOU enter both an email address and a password! 
Now You can blah, blah, blah 











Co back and try again 











图 8-19 ”成 功 登 录 后 ， 用 户 会 看 到 这 样 的 消息 图 8-20 ”如 果 没 有 提交 Email 地 址 或 密码 ， 则 会 看 到 
这 样 的 消息 








Logm Form 


Users who are logged in can take advantage of certain features like this, 


that, and the other thing 


The submitted email address and password do not match those on filel 








Co back and try again 








图 8-21 ”如 果 与 脚本 中 的 Email 地 址 或 密码 不 匹配 ， 则 用 户 会 看 到 这 样 的 错误 消息 


w 提示 

口 在 实际 使 用 当中 ， 可 能 需要 为 错误 消息 添加 一 些 CSS 样 式 ， 使 其 更 加 突出 。 本 章 的 8.6 市 

将 介绍 这 一 功能 。 

口 这 里 用 于 判断 输入 框 是 否 存在 的 小 技巧 可 能 让 人 迷惑 。 它 的 工作 原理 是 因为 同样 的 脚 
本 一 一 1ogin.php 一 一 会 被 用 户 访问 两 次 。 第 一 次 访问 时 表单 并 没有 被 提交 ， 所 以 主 条 
件 会 是 FALSE， 然 后 显示 表单 。 在 用 户 单 击 submit 之 后 ， 会 再 次 访问 页 面 ， 此 时 条 件 会 变 














为 TRUE。 
口 如 果 希 望 页 面 在 处 理 完 表单 之 后 ， 能 够 立即 再 次 显示 表单 ， 可 以 这 样 做 : 
if ($_SERVER['REQUEST METHOD'] == 'POST') { 
// 处 理 表 单 。 
} 


// 显示 表单 。 
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8.6 ”使 表单 更 具 粘 性 


粘性 表单 (sticky form) 会 记 住 输入 到 其 中 的 值 。 和 
一 个 常见 的 示例 就 是 搜索 引擎 ， 它 总 会 在 搜索 框 中 显 “| Reglstratton orm 

示 所 输入 的 关键 词 ， 其 至 在 显示 搜索 结果 时 亦 是 如 此 。 | ee ee en 
当 用 户 由 于 输入 错误 值 而 没 能 准确 完成 表单 ， 从 而 需 

要 重新 提交 时 ， 可 能 就 需要 使 用 粘性 表单 了 (参见 图 
8-22) 。 SANE ep 

从 技术 的 角度 来 看 ， 粘 性 表单 可 以 通过 为 表单 元 。 | so 

素 提 供 预 设 值 来 实现 。 可 以 在 创建 表单 时 为 输入 框 设 























置 value 输 入 框 : 
First N Larry 
<input type="text" name= 
"first name" value="Stephanie" /> Last Name 





要 使 用 PHP 预 设 值 ， 可 以 在 引号 中 打印 适当 的 变 











| 





8-22 ”在 没有 完成 表单 提交 时 ， 粘 性 表 
单 可 以 帮 用 户 找到 漏 填 的 表单 值 





ba 
































<input type="text" name="first _ name" 
value="<?php print $_POST ['first name']; ?>" /> 


表单 第 一 次 运行 时 ，PHP 代 码 什 么 都 不 打印 (因为 变量 没有 值 )。 如 果 表 单 在 提交 后 再 次 显 
示 ， 则 用 户 之 前 在 表单 中 输入 的 值 会 自动 显示 出 来 。 这 是 基本 的 方法 , 但 更 加 专业 的 实现 方式 需 
要 处 理 两 件 事 …… 

首先 ， 最 好 不 要 引用 不 存在 的 变量 。 这 样 会 引发 PHP 警 告 ， 而 如 果 将 PHP 代 码 怠 入 到 一 个 表 
单元 素 属性 中 ,警告 代码 只 能 出 现在 HTML 的 源 代码 中 (参见 图 8-23)。 为 了 避免 这 一 点 , 需要 在 
打印 之 前 检查 变量 是 否 已 被 设置 。 


<input type="text" name="first name" value="<?php if (isset($_ POST 
*['first name']) { print $_ POST['first name']; } ?>" /> 



































外 
T 
的 
pl 
N 
m 
外 
吕 
m 
外 


<p>First Name: <input type="text" name="first name"™ 
b>No es Undefined index: first name in <b>/Users/larryullman/Sites/phpvds4/register .php</b> 


"jp> 

















图 8-23 ”HTML 源 代码 ， 显 示 由 于 引用 不 存在 变量 而 产生 的 PHP 页 面 错误 


其 次 ， 用 户 提 交 的 值 中 的 某 些 字符 ， 直 接 打印 在 表单 元 素 的 value 属 性 中 可 能 会 引发 问题 。 
要 避免 这 样 的 问题 ， 需 要 使 用 htmlspecialchars () 函数 (在 第 5 章 中 有 所 讨论 ) 。 考 虑 到 这 一 
点 ， 下 面 给 出 了 这 段 代码 更 长 但 却 是 更 好 的 版 本 : 


<input type= "text" name="first name" value="<?php if (isset($_POST 
>['first name']) { print htmlspecialchars($_ POST['first name']); } ?>" /> 


作为 演示 ， 我 们 来 创建 注册 表单 的 界面 (参见 图 8-24)。 
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Registration Form 


so that YOU can take advantage of certain features like this, that, 








nther thing 
First Name 
Last Name 


Email Address: | 


Confirm Password 


图 8-24 ”用户 首次 看 到 的 注册 表 自 




















这 使 表单 更 具 粘 性 的 步 又 
(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 PHP 脚 本 ， 命 名 为 register .php (参看 脚本 8-9): 


<?php // 脚本 8-9 - register.php 








脚本 8-9 ”注册 表单 使 用 了 所 谓 的 粘性 特征 ， 这 样 它 就 可 以 回 显 用 户 输入 的 值 


<?php // 脚本 8-9 - register.php 
/* 页 面 可 以 用 于 用 户 登 录 (理论 上 )。*/ 


// 设置 页 面 标题 并 包含 头 文件 : 

define('TITLE', 'Register'); Se 
include('templates/header.html'); 

// 打印 一 些 介 绍 文本 : 

print '<h2>Registration Form</h2> 








\D own 心 wN 


























10 <p>Register so that you can take advantage of certain features like this, that, 
and the other thing.</p>'; 

11 

12 // 添加 CSS: 

13 Print '<style type="text/css" media="screen"> 

14 .error { color: red; } 

15 </style>'; 

16 

17 // 检查 表单 是 否 已 提交 : 

18 if ($_SERVER['REQUEST METHOD'] == 'POST') { 

19 

2.0 $Sproblem = FALSE; // 目前 一 切 正 常 。 

21 

22 // 检查 每 个 值 …… 

23 if (empty($_POST['first name'])) { 

24 sproblem = TRUE; 

25 print '<p class="error">Please enter your first name!</p>'; 
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27 

28 if (empty($_POST['last name'])) { 

29 $sproblem = TRUE; 

30 print '<p class="error">Please enter your last name!</p>'; 

31 } 

32 

33 if (empty($_ POST['email'])) { 

34 $sproblem = TRUE; 

25 print '<p class="error">Please enter your email address!</p>'; 

36 } 

37 

38 if (empty($_POST['password1'])) { 

39 $sproblem = TRUE; 

40 print '<p class="error">Please enter a password!</p>'; 

41 } 

42 

43 If ($_POST['password1l'] != $_POST['password2']) { 

44 $sproblem = TRUE; 

45 print '<p class="error">Your password did not match your confirmed 

password!</p>'; 

46 } 

47 

48 if (!Sproblem) { // 判断 是 否 有 错误 发 生 …… 

49 

50 // 打印 一 条 消息 : 

5 print '<p>You are now registered! 

<br/>Okay, you are not really registered but...</p>'; 

52 

53 // 清除 数组 中 的 元 素 : 

54 $_POST = array(); 

55 

56 ) else { // 表单 未 填 完 整 。 

3 

58 print '<p class="error">Please try again!</p>'; 

59 

60 } 

61 

62 } // 结束 处 理 表 单 的 条 件 语句。 

63 

64 // 创建 表单 : 

G5 ?和 

66 <form action="register.php" method="post"> 

67 

68 <p>First Name: <input type="text" name="first name" size="20" 
value="<?php if (isset($ POST ['first name'])) { print 
htmlspecialchars($ POST['first name']); } ?>" /></p> 

69 

70 <p>Last Name: <input type="text" name="last name" size="20" 
value="<?php if (isset($ POST['last name'])) { print 
htmlspecialchars($ POST['last name']); } ?>" /></p> 

71 

72 <p>Email Address: <input type="text" name="email" 


size="20" value="<?php if (isset($ POST['email'])) { print 
htmlspecialchars($ POST['email']); } ?>" /></p> 
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73 
74 <p>Password: <input type="password" name="passwordl1" 
size="20" value="<?php if (isset($ POST['passwordl1'])) 
{ print htmlspecialchars($ POST['password1']); } ?>" /></p> 
75 <p>Confirm Password: <input type="password" name="password2" 
size="20" value="<?php if (isset($ POST['password2'])) 
{ print htmlspecialchars($ POST['password2']); } ?>" /></p> 


76 

EY <p><input type="submit" name="submit" value="Register!" /></p> 
78 

79 </form> 

80 

81 <?php include('templates/footer.html'); // Need the footer. ?> 
(2) 设置 页 面 标题 并 包含 HIML 页 头 : 

define('TITLE', 'Register'); 


include('templates/header.html'); 
(3) 添加 一 些 介绍 性 文本 并 定义 一 个 CSS 类 : 


print '<h2>Registration Form</h2> 
<p>Register so that you can take advantage of certain 
"features like this, that, and the other thing.</p>'; 
print '<style type="text/css" media="screen"> 
.error { color: red; } 
</style>'; 


如 果 没 能 正确 完成 注册 表单 ， 为 了 更 加 突出 生成 的 错误 消息 ， 这 里 定义 了 一 个 CSS 类 ， 它 会 
使 文字 的 颜色 变 为 红色 。 尽 管 CSS 通 常 定义 在 页 面 的 顶部 ， 但 实际 上 可 以 将 其 放 在 任何 位 置 。 

(4) 检查 表单 是 否 已 经 被 提交 

if (S$_SERVER['REQUEST METHOD'] == 'POST') { 

和 登录 页 一 样 ,该 脚本 同时 显示 和 处 理 注册 表单 。 检 查 表单 是 否 已 经 提交 ,这 里 使 用 的 代码 
与 前 面 示例 中 的 一 样 。 

(5) 创建 一 个 标志 变量 : 

$sproblem = FALSE; 


$sproblem 变 量 用 于 指出 是 否 出 现 错误 。 在 正式 注册 用 户 之 前 ， 需 要 确保 每 个 表单 输入 框 都 
已 完全 填写 。 在 一 开始 ， 该 变量 被 设置 为 FALSE， 因 为 尚 无 错误 发 生 。 
(6) 检查 输入 的 first_name: 
if (empty($_POST['first name'])) { 
$problem = TRUE; 


print '<p class="error">Please enter your first name!</p>'; 


} 

检查 用 户 是 否 输入 了 first_name 值 是 一 个 非常 简单 的 测试 ， 只 需 判断 该 变量 是 否 为 非 空 。 
(该 技术 在 第 6 章 中 首次 介绍 。) 如 果 该 变量 是 空 ， 则 通过 将 标志 变量 设置 为 TRUE 来 指出 错误 ,并 
打印 一 条 错误 消息 。 错 误 消息 的 class 值 是 error， 因 此 会 向 其 应 用 CSS 样 式 。 




















上 
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(7) 对 Last_name 和 Email 地 址 重复 这 个 验证 ， 


if (empty($_POST['last name'])) { 
sproblem = TRUE; 
print '<p class="error">Please enter your last name!</p>'; 
} 
if (empty($_POST['email'])) { 
sproblem = TRUE; 
print '<p class="error">Please enter your email address!</p>'; 


} 
这 两 个 检查 是 对 用 户 名 的 检查 代码 进行 修改 后 得 到 的 。 








(8) 验证 密码 : 
if (empty($_POST['password1'])) { 


sproblem = TRUE; 
print '<p class="error">Please enter a password!</p>'; 
} 
if ($_POST['password1'] != $_POST['password2']) { 
sproblem = TRUE; 
print '<p class="error">Your password did not match your confirmed password!</p>'; 


} 

密码 的 验证 需要 两 个 条 件 语 句 。 第 一 个 条 件 检查 了 $_POST['password1'] 变 量 是 否 为 空 。 
第 二 个 条 件 检 查 了 $_POST['password1'] 变 量 是 否 等 于 $_POST['password2'] 变 量 。 这 里 无 
需 检查 $_POST['password2'] 是 否 为 空 ， 因 为 如 果 它 为 空 而 $_POST[ 'passwordl' ] 不 为 空 ， 
则 第 三 个 条 件 会 捕获 这 个 问题 。 如 果 $_POSTI['password2'] 和 $_POST['password1'] 同 时 为 
空 ， 则 第 一 个 条 件 会 捕获 这 种 问题 。 

(9) 检查 是 否 有 错误 发 生 : 

if (!$problem) { 


print '<p>You are now registered!<br/>Okay, you are not really registered but...</p>'; 
$_POST = array(); 


如 果 没 有 错误 , 则 $problem 变 量 仍然 为 FALSE, 该 代码 第 一 行 中 的 条 件 将 为 TRUE (这 是 因 
为 $problem 的 值 为 FALSE)。 在 这 种 情况 下 , 注册 流程 将 得 以 执行 。 正式 的 注册 流程 需要 将 数据 
存放 在 文件 或 数据 库 中 ， 这 里 还 没有 开发 这 些 代码 ， 所 以 只 是 用 一 个 简单 的 消息 来 替代 。 

接 下 来 ， 用 array () 为 $_POST 变 量 赋值 。 这 一 行 代码 的 效果 是 擦 除了 $_POST 变 量 中 的 内 容 
(也 就 是 说 ,将 其 重 置 为 一 个 空 数组 )。 只 在 注册 成 功 (理论 上 ) 之 后 执行 这 一 步 ， 这样 输 入 的 值 
就 不 会 再 次 显示 在 注册 表单 中 (参见 第 (12) 步 )。 

(10) 完成 条 件 语 句 : 

} else { // 表单 未 填 完 整 。 
print '<p class="error">Please try again!</p>'; 


} 
】 // 结束 处 理 表单 的 条 件 语句 。 


如 果 发 生 了 错误 则 会 使 用 else 子 句 ， 要 求 用 户 重新 完成 表单 。 
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(11) 打开 HTML 表单 


<form action="register.php" method="post"> 


和 登录 示例 不 同 ,该 页 面 总 是 显示 表单 。 因 此 ， 表 单 并 不 属于 条 件 语 句 的 一 部 分 。 另 外 ， 由 
于 要 生成 大 量 的 HITML， 所 以 暂时 离开 PHP 节 ， 直 接 输出 HTML 会 比较 方便 一 些 。 
(12) 创建 粘性 的 first name 输 入 框 : 


<p>First Name: <input type="text" name="first name" size="20" 
*value="<?php if (isset($_ POST['first name'])) { print 
htmlspecialchars($_POST['first name']); } ?>" /></p> 


为 了 让 first_name 输 入 具有 粘性 ,需要 通过 打印 $_POST['first_name'] 变 量 的 值 来 预 设 
其 value 属 性 ， 但 只 在 已 设置 该 变量 的 值 时 这 样 做 。 因 此 ， 需 要 将 这 个 条 件 放 到 HTML 表 单元 素 
value 节 中 的 PHP 标 签 里 。 如 前 所 述 ， 需 要 使 用 htmlspecialchars () 函数 来 处 理 可 能 存在 问题 
注意 ， 如 果 用 户 正 确 填写 完 表单 ，$_POST 数 组 会 被 重 置 ， 使 PHP 的 条 件 语 句 为 FALSE。 
(13) 针对 last_name 和 Email 地 址 重复 这 一 过 程 : 


<p>Last Name: <input type="text" name="last name" size="20" 
一 Value="<?php if (isset($_ POST['last name'])) { print 
"htmlspecialchars($_POST['last name']); } ?>" /></p> 
<p>Email Address: <input 

















"type="text" name="email" size="20" value="<?php if 
(isset($_POST['email'])) { print htmlspecialchars($_POST 
*['email']); } ?>" /></p> 
这 些 代码 是 从 第 (12) 步 中 演变 过 来 的 ， 适 当地 修改 了 变量 名 。 
(14) 添加 表单 的 剩余 部 分 : 
<p>Password: <input type="password" name="passwordl1" size="20" value="<?php if 
» (isset($_POST['password1'])) { print htmlspecialchars 
»($_POST['password1']); } ?>" /></p> 
<p>Confirm Password: <input type="password" name="password2" 
—>size="20" value="<?php if (isset($ POST['password2'])) 
一 { print htmlspecialchars($_ POST['password2']); } 
”> D> 
<p><input type="submit" name="submit" value="Register!" /></p> 
</form> 


以 前 不 能 为 password 类 型 的 输入 框 设置 预 设 值 ， 但 现在 一 些 浏 览 器 已 支持 这 个 功能 。 接 下 
来 放置 了 提交 按钮 和 form 关 闭 标签 。 

(15) 完成 PHP 页 面 : 

<?php include('templates/footer.html'); ?> 

最 后 一 步 是 包含 HIML 页 脚 文件 。 

(16) 将 文件 保存 为 register .php， 放 置 在 启用 了 PHP 的 服务 器 上 的 适当 目录 中 ， 并 在 Web 
浏览 器 中 进行 测试 (参见 图 8-25 和 图 8-26)。 
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Registration Form 





YOU can take advantage of certain features like this, that, 





ther thing 





Raise High the Roof Beam! 


First Name: |Larry 
Favorite qd 


Last Name Regi stration Form 


l don't exactly ki 





you can take advantage of certain features like this, that, - The Catcher i 


Email Address: |me(@example.com 





Confirm dhe 



































图 8-25 注册 表单 指出 了 发 生 的 错误 ， 并 图 8-26 ”用 户 填 写成 功 之 后 的 注册 表单 











保留 了 除 密码 之 外 的 表单 值 


V 提示 

口 根据 (X)HTML 标 准 ， 必 须 将 表单 输入 框 的 属性 放 在 引号 中 ， 特 别 是 要 放 在 双 引 号 中 。 如 
果 没 有 用 引号 把 值 引 起 来 , 则 其 中 任何 一 个 空格 都 会 标志 着 值 的 结束 (例如 , Larry Ullman 
只 会 在 表单 输入 框 中 显示 出 Larry)。 虽 然 在 HIML5 中 ， 输 入 框 的 属性 不 必 再 用 引号 括 起 
来 ， 但 我 建议 你 加 上 引号 。 

口 要 为 单 选 按 钮 或 复 选 框 设置 预 设 值 ， 应 该 在 其 文本 框 标签 中 添加 checked="checked": 
<input type="checkbox" name="interests[ ]" value="Skiing" checked="checked" /> 
当然 ， 需 要 使 用 一 个 PHP 条 件 语句 ， 来 确定 是 否 应 该 将 这 些 文字 添加 到 元 素 的 定义 中 。 

口 要 为 下 拉 框 设置 预选 值 ， 请 使 用 selected="selected": 


<select name="year"> 

<option value="2011">2011</option> 

<option value="2012" selected="selected">2012</option> 
</select> 


同样 ， 需 要 使 用 一 个 PHP 条 件 语 句 来 检查 是 否 需 要 将 这 些 文字 添加 到 元 素 定义 中 。 
口 要 为 文本 区 设置 预 设 值 ， 需 要 将 值 放置 在 textarea 标 签 之 内 : 


<textarea name="comments" rows="10" cols="50">preset value</textarea> 
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8.7 发 送 Email 


从 理论 上 说 , 使 用 PHP 发 送 Email 非 常 简单 ， 只 需要 使 用 mail () 函数 即 可 。mai1 () 国 数 使 用 
服务 器 上 的 Email 应 用 程序 (如 Unix 或 Mac OS X 上 的 sendmai1) 或 MTP (Simple Mail Transfer 
Protocol) 服务 器 来 向 外 发 送 消息 。 该 函数 的 基本 使 用 方法 是 : 

mail(to, subject, body); 

第 一 个 参数 是 Email 应 该 被 发 送 的 地 址 〈 可 以 是 多 个 地 址 ， 使 用 逗号 分 隔 ) 。 第 二 个 参数 是 消 
上 息 的 主题 ， 第 三 个 参数 是 消息 的 内 容 。 

该 函数 还 可 以 接受 另外 一 个 参数 ， 用 于 为 Email 添 加 更 多 详细 信息 (附加 头 )， 包括 发 件 人 地 
址 、Email 优 先 级 和 抄 送 地 址 : 


mail('someone@example.com', 'Test Email', 'This is a test email', 'From: 
»'email@example.com'); 


尽管 这 从 理论 上 说 很 容易 ， 但 在 显示 代码 中 实际 使 用 该 函数 时 会 更 为 复杂 。 对 于 初学 者 来 
说 ,设置 自己 的 计算 机 来 发 送 Email 是 很 难 的 (请 参见 框 注 “ 配 置 服务 器 以 发 送 Email”)。 
其 次 ， 需 要 执行 一 些 步骤 来 防止 恶意 用 户 试图 用 表单 来 发 送 垃圾 邮件 。 在 下 一 个 示例 中 , 会 
将 一 封 Email 发 送 到 用 户 提 交 的 Email 地 址 。 这 里 允许 用 户 提交 多 个 Email 地 址 (参见 图 8-27)， 邮 
件 会 分 发 到 每 个 Email 地 址 。 有 很 多 种 安全 防护 措施 。 本 书 采 用 一 种 相对 简单 的 方法 ， 即 确认 提 
供 的 一 个 Email 地 址 中 只 包含 一 个 “@ ”标志 。 可 以 使 用 substr_count () 函数 计算 字符 串 中 子 
字符 串 的 数量 ; 


if (substr_count ($_POST['email']，'Q@') == 1) {... 




















Email Address 


me®@example.com,you@example.eduwhomewver(@example.net 


























图 8-27 用 户 可 以 使 用 类 似 的 表单 将 邮件 发 送 给 多 个 收 件 人 








配置 服务 器 以 发 送 Email 
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DDNU We O000000 Unixd0U00000Linvxd O00U00 
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UUOUUUO0O0U0O0UO0O0OU0U0O0000PHPEUOOOOOU0O0D0 EmailiUOUOO0OOD 
UOUOUUUOUU0O0O00Yweou0OO0OuUu00o00000000000U0000000 
UOUOUUUOUU0O0O0D0PHPEUUOUOOOO0OOO0ODU Email 


下 面 来 向 注册 页 面 中 添加 一 个 对 mail () 函数 的 调用 ， 这 样 你 就 能 对 该 函数 的 使 用 方式 有 一 
个 感性 的 认识 。 


信使 用 PHP 发 送 邮 件 








(1) 在 文本 编辑 器 或 IDE 中 打开 register.php (参看 脚本 8-9 ) 。 

(2) 修改 Email 验 证 ， 检 查 Email 地 址 中 是 否 只 有 一 个 “@” 标 志 (参看 脚本 8-10) : 

if (empty($_POST['email']) || (substr_count($_POST['email'],'@') != 1) ) { 

如 果 地 址 为 空 或 “@” 标 志 的 个 数 不 是 1 的 话 ， a 这 个 验证 并 不 完美 (还 
差 得 很 远 ) ， 但 是 这 已 经 让 使 用 Email 地 址 的 安全 风险 变 得 小 多 了 。 参 见 本 节 后 面 的 提示 ， 改 进 这 
个 验证 。 


脚本 8-10 在 PHP 中 ， 可 以 通过 mail() 函数 来 发 送 Email 





<?php // 脚本 8-10 - register.php #2 
/* 页 面 可 以 用 于 用 户 登 录 (理论 上 )。*/ 


J 
2 
3 
4  // 设置 页 面 标题 并 包含 头 文件 : 

于 define('TITLE', 'Register'); 

6 include('templates/header.html'); 
2 
8 
9 


// 打印 一 些 介绍 文本 : 
print '<h2>Registration Form</h2> 














10 <p>Register so that you can take advantage of certain features like this, 
that, and the other thing.</p>'; 

11 

12 // 添加 CSS: 

13 Print '<style type="text/css" media="screen"> 

14 .error { color: red; } 

15 </style>'; 

16 

17 // 检查 表单 是 否 已 提交 : 

18 if ($_SERVER['REQUEST _ METHOD'] == 'POST') { 

19 

20 Sproblem = FALSE; // 目前 一 切 正常 。 

21 

22 // 检查 每 个 值 …… 

| if (empty($_POST['first name'])) { 

24 $sproblem = TRUE; 

25 print '<p class="error">Please enter your first name!</p>'; 

26 } 

27 

28 if (empty($_POST['last name'])) { 





29 $sproblem = TRUE; 
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print '<p class="error">Please enter your last name!</p>'; 


if (empty($_ POST['email']) || (substr count($_ POST['email'], '@') != 1) ) { 
sproblem = TRUE; 
print '<p class="error">Please enter your email address!</p>'; 





if (empty($_POST['password1'])) { 
$sproblem = TRUE; 
print '<p class="error">Please enter a password!</p>'; 





If ($_POST['password1l'] != $_POST['password2']) { 
sproblem = TRUE; 
print '<p class="error">Your password did not match your confirmed 
password!</p>'; 





if (!1$problem) { // 判断 是 否 有 错误 发 生 …… 


// 打印 一 条 消息 : 
print '<p>You are now registered!<br/>Okay, you are not really registered 
DU AD 


// 发 送 Email: 

$body = "Thank you for registering with the J.D. Salinger fan club! Your password 
is '{$ POST['password1']}'."; 

mail($ POST['email'], 'Registration Confirmation', $body, 'From: 
admin@example.com'); 


// 清除 数组 中 的 元 素 : So 


$_POST = array(); 





} else { // 表单 未 填 完 整 。 


print '<p class="error">Please try again!</p>'; 


】 // 结束 处 理 表单 的 条 件 语 向。 


// 创建 表单 : 


3 


<form action="register.php" method="post"> 


<p>First Name: <input type="text" name="first name" size="20" value="<?php if 
(isset($_POST['first name'])) { print htmlspecialchars($_ POST['first 
_name']); } ?>" /></p> 


<p>Last Name: <input type="text" name="last name" size="20" value="<?php if 
(isset($_POST['last name'])) { print htmlspecialchars($_POST['last name']); 
a 
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76 <p>Email Address: <input type="text" name="email" size="20" value="<?php if 
(isset($_ POST['email'])) { print htmlspecialchars($_ POST['email']); } ?>" /> 
</p> 

a 

78 <p>Password: <input type="password" name="passwordl1" size="20" value="<?php if 


(isset($_ POST['password1'])) { print htmlspecialchars($_ POST['password1l']); } 
a 








79 <p>Confirm Password: <input type="password" name="password2" size="20" value= 
"<?php if (isset($ POST['password2'])) { print htmlspecialchars($_POST['password2']); 
于 是 人 RD> 

80 

81 <p><input type="submit" name="submit" value="Register!" /></p> 

82 

83 </form> 

84 

85 <?php include('templates/footer.html'); // 包含 页 脚 文件 。? 

(3) 在 注册 消息 之 后 (参见 第 51 行 )， 添 加 下 面 的 代码 : 

Sbodqy = "Thank you for registering with the J.D. Salinger fan 

Club! Your password is '{$_POST['password1']}'."; 
mail($_POST['email'], 'Registration Confirmation', S$body, 'From: 


"admin@example.com’'); 


有 时 使 用 该 函数 最 简单 的 方式 ， 是 将 正文 建立 在 一 个 变量 中 ， 然 后 传递 给 mai1 () 函数 。 消 
息 本 身 将 被 发 送 到 用 户 注册 时 提供 的 地 址 中 ， 标 题 为 Registration Confirmation ， 来 自 
admin@example.com。 如 果 在 一 台 真 实 的 服务 器 上 运行 该 程序 ， 应 该 在 FROM 值 中 使 用 该 站 点 的 
真实 Email 地 址 。 

(4) 保存 文件 ， 将 其 放置 在 支持 PHP 和 Email 的 服务 器 上 的 适当 目录 中 ， 并 在 Web 浏 览 器 中 进 
行 测 试 (参见 图 8-28)。 





























Registration Form 


Register so that you can take advantage of certain features like this, that, 





and the other thing 


First Name Larry 


Last Name: |UllIman 


Email Address: Imet@example.com,you(@example.edu,whor 














图 8-28 ”再 次 测试 注册 表单 
(5) 填写 完 表单 ， 检 查 邮箱 中 的 消息 (参见 图 8-29)。 
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AAMA ,| Registration Confirmation 一 Inbox = 





From: admin@example.com 
Subject Registration Confimation 
Date: December 13, 2010 9:09:05 PM EST 
To lany@lamyullIman.com 


Thank you for registering with the J.D. Salinger fan club! Your 


| 

| 

| 

| 

password is ‘superSECRET.. | 

















图 8-29 ”这 是 当 伪 注册 过 程 成 功 后 ， 由 PHP 脚 本 发 送 的 Email 





提示 

口 8.10 节 会 介绍 一 个 非常 好 的 验证 Email 地 址 的 工具 ， 前 提 是 PHP 必 须 是 5.2 及 以 上 版 本 。 

口 在 我 编写 的 《PHP 6 与 MySQL 5 基础 教程 》 一 书 以 及 本 书 的 在 线 论 坛 (www.LarryUllman. 

com/forum/) 中 ， 讨 论 了 通过 PHP 脚 本 确保 Email 发 送 成 功 的 其 他 方法 。 

D 如 果 在 接收 由 PHP 发 送 的 Email 时 遇 到 问题 ， 首 先 请 确保 在 不 使 用 PHP 的 情况 下 ， 邮 件 服 
务 器 能 够 正常 工作 。 然 后 确认 使 用 了 有 效 的 FROM 地 址 。 最 后 ,尝试 使 用 一 个 不 同 的 接收 
地 址 ， 并 查看 一 下 垃圾 邮件 ， 看 看 消息 是 否 被 放 到 这 里 了 (如 果 可 以 的 话 )。 

口 还 可 以 发 送 带 有 附件 的 Email 或 HTML 格 式 的 Email, 但 这 样 做 需要 更 多 复杂 的 编码 (通常 
会 引入 类 和 对 象 )。 幸 运 的 是 ,很 多 程序 员 已 经 开发 了 可 以 使 用 的 解决 方案 。 请 参见 附录 
B， 其 中 给 出 的 Web 站 点 可 能 会 有 所 帮助 。 

口 mail () 函数 会 返回 一 个 值 (1 或 0) 表示 它 是 否 执行 成 功 。 这 个 值 只 能 表示 PHP 是 否 能 够 
尝试 发 送 Email (通过 判断 是 否 安装 有 Email 系统 )。 没 有 哪 种 简单 的 方法 能 够 使 用 PHP 判 
断 一 个 Email 地 址 是 否 有 效 ， 或 最 终 用 户 是 否 收 到 了 消息 。 

口 要 向 多 个 地 址 发 送 一 封 Email， 可 以 使 用 CC 参数 ， 或 者 用 喜 号 分 隔 TO 参 数 中 的 地 址 。 = 

口 要 在 Email 正 文中 创建 新 行 ， 可 以 在 创建 消息 时 使 用 多 行文 本 ,或 者 在 双 引 号 中 使 用 换行 

符 (\n)。 

口 如 果 除 了 FROM 地 址 之 外 ， 还 想 发 送 多 个 头 ， 可 以 使 用 \r\n 组 合 将 它们 分 开 : 


mail ('email@example.com','Testing', S$body, "From:email@example.org 
"\r\nBcc:hidden@example.net,third@example.com"); 


8.8 输出 缓冲 


本 章 和 后 面 章 闻 中 用 到 的 某 些 函 数 ， 只 能 在 没有 任何 东西 被 发 送 到 浏览 器 之 前 调用 。 这些 函 
数 包 括 header () 、setcookie() 和 session_start()。 如 果 在 Web 浏 览 器 已 经 收 到 了 一 些 文 
本 、HTML 或 哪怕 是 一 个 空格 之 后 调用 这 些 函 数 ， 就 会 得 到 一 个 恼人 的 HTTP 头 已 发 送 错误 消息 
(参见 图 8-30)。 
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Warning: Cannot modify header information - headers already sent by loutput started 
at JUsersilarryullmanisSites/!phpvgqs4 /templates /header .html:4) in Usersilarryullman 
lSites/phpvqs41login.php on line 22 

















图 8-30 ”如 果 在 调用 header () 之 前 浏览 器 接收 到 了 任何 HTML， 都 会 看 到 这 样 的 错误 消息 


本 书 为 初学 PHP 的 开发 者 提供 了 一 种 解决 方法 ， 即 采用 输出 绥 冲 〈output buffering， 或 称 输 
出 控制 )。 在 一 般 的 PHP 脚 本 中 ，PHP 标 签 之 外 的 任何 HTML 都 会 立即 发 送 到 Web 浏 览 器 ,一 量 执 
行 了 print 语 句 ， 所 有 的 打印 内 容 也 是 如 此 。 利 用 输出 缓冲 ，HIML 和 打印 的 数据 (输出 ) 将 被 
放 到 缓冲 (也 就 是 内 存 ) 中 。 当 脚本 执行 结束 后 ， 缓 冲 将 被 发 送 到 Web 浏 览 器 ， 或 者 如 果 需 要 的 
话 ， 缓冲 可 以 清空 而 不 发 送 到 Web 浏 览 器 。 使 用 输出 缓冲 的 原因 有 很 多 ,但 对 于 初学 者 来 说 ,一 
个 主要 的 优点 在 于 ， 使 用 这 些 函 数 时 ， 无 需 担 心 会 出 现 “HTTP 头 已 发 送 ” 的 错误 。 尽 管 还 没有 
用 到 这 些 函 数 ， 但 本 章 还 是 在 这 里 简要 介绍 了 输出 缓冲 ， 因 为 它 极 大 程度 地 减少 了 在 使 用 HTTP 
头 〈8.9 节 介绍 ) 、cookie (第 9 章 介绍 ) 和 session (第 9 章 介 绍 ) 时 的 错误 。 

要 启用 输出 缓冲 ， 请 在 页 面 的 最 顶端 使 用 ob_start () 函数 。 一 但 调用 了 该 函数 ， 则 每 个 
print 和 类 似 的 函数 都 会 将 数据 发 送 到 内 存 缓冲 中 ,而 不 是 发 送 到 Web 浏 览 器 。 相反 ,HTTP 调 用 
(如 header () 和 setcookie()) 不 会 缓冲 ， 而 是 按照 常规 方式 处 理 。 

在 脚本 的 结尾 处 , 调用 ob_endq_flush () 函数 , 将 积累 下 来 的 缓冲 发 送 到 Web 浏 览 器 。 或 者 ， 
可 以 使 用 ob_eng_clean () 函数 删除 缓冲 的 数据 而 不 进行 传输 。 这 两 个 国 数 都 会 为 当前 脚本 关闭 
输出 缓冲 。 

从 程序 员 的 视角 来 看 ， 输 出 缓冲 人 允许 以 更 加 线性 的 方式 来 组 织 脚 本 ， 而 不 用 关心 HITP 头 。 
下 面 我 们 来 修改 一 下 header .html 和 footer .html， 这 样 每 个 页 面 都 能 使 用 输出 缓冲 。 目 前 你 
可 能 不 会 从 中 受益 , 但 在 以 后 编写 复杂 的 程序 时 ， 它 可 能 会 为 你 减少 很 多 出 错 的 机 会 ,使 你 保持 
清醒 的 头脑 。 


之 使 用 输出 缓冲 


























(1) 在 文本 编辑 器 或 IDE 中 打开 header .html (参看 脚本 8-6)。 
(2) 在 页 面 的 最 顶端 、 任 何 HTML 代 码 之 前 ， 添 加 下 面 的 代码 (参看 肢 本 8-11) : 
<?php 


ob_start(); 
Pe 





脚本 8-11 将 ob_start () 函数 放 在 header html 脚 本 的 最 顶端 ， 为 Web 应 用 程序 添加 答 出 缓冲 


<?php // 脚本 8-11 - header.html #4 


工 
2 
3 ”// 开局 输出 缓冲 : 
4 ob _ start(); 
8 
6 


?><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" 
"http://www.w3.o0org/TR/xhtml11/DTD/xhtml11 .dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en"> 
8 <head> 


-J 
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9 <title><?php // 打印 页 面 标题 。 

10 if (defined('TITLE')) { // 检查 标题 是 否 已 定义 。 

的 Print TITLE; 

12 } else { // 标题 没有 定义 。 

13 print 'Raise High the Roof Beam! A J.D. Salinger Fan Club'; 

14 } 

5 ?></title> 

16 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 

43 <link rel="stylesheet" href="css/1.css" type="text/css" 

media="screen,projection" /> 

18 </head> 

19 <body> 

20 <div id="wrapper"> 

2 

22 <div id="header"> 

23 <p class="description">A J.D. Salinger Fan Club</p> 

24 <hl><a href="index.php">Raise High the Roof Beam!</a></hl1l> 

和 5 <ul id="nav"> 

26 <li><a href="books.php">Books</a></1i> 

27 <l1i><a href="#">Stories</a></1i> 

28 <li><a href="#">Quotes</a></1i> 

29 <li><a href="login.php">Login</a></1i> 

30 <li><a href="register.php">Register</a></1i> 

3 </ul> 

Ke </div><!-- 页 头 --> 

33 

34 <div id="sidebar"> 

385, <h2>Favorite Quotes</h2> 

36 <p class="news">I don't exactly know what I mean by that, but I mean it. 
<br/>- <em>The Catcher in the Rye</em></p> 

Ey <p class="news">I privately say to you, old friend... please accept from me 
this unpretentious bouquet of early-blooming parentheses: (((()))).<br/>- 
<em>Raise High the Roof Beam, Carpenters and Seymour: An Introduction</em> 
</p> 

38 <p><?php // 打印 当前 日 期 和 时 间 …… 

39 // 设置 时 区 : 

40 date_default_ timezone_set('America/New_York'); 

41 

42 // 打印 日 期 和 时 间 : 

43 Deint date(Tgogei lL RY 

44 ?></p> 

45 </div><!-- 边栏 --> 

46 

47 <div id="content"> 

48 <!-- 可 变 内容 开 始 。--> 





使 用 输出 缓冲 的 关键 在 于 ， 要 在 脚本 中 尽 可 能 早 地 调用 ob_start () 函数 。 这 个 示例 中 ,在 
任何 HTML 之 前 创建 了 一 个 特殊 的 PHP 闻 ， 并 在 这 里 调用 了 ob_start () 。 在 页 头 文件 中 打开 输 
出 缓冲 , 并 在 页 脚 文件 中 将 其 关闭 , 这 样 做 可 以 让 Web 应 用 程序 中 的 每 个 页 面 都 能 使 用 输出 缓冲 。 

(3) 在 文本 编辑 器 或 IDE 中 打开 footer .html (参看 脚本 8-8)。 

(4) 在 脚本 的 结束 位 置 ， 所 有 的 HTML 之 后 ,添加 (参看 脚本 8-12): 
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<?php 
ob_engd flush(); 


时 加 


脚本 8-12 在 页 脚 文件 的 结尾 处 使 用 ob_end_flush () 完成 了 输出 缓冲 ， 这 会 将 积累 下 来 的 缓 
冲 发 送 到 Web 浏 览 器 


1 <!-- 可 变 内 容 结束 。--> 

2 </div><!-- 内 容 --> 

3 

4 <div id="footer"> 

5 <p>Template design by <a href="http://www.sixshootermedia.com"> 
Six Shooter Media</a>.</p> 

6 <p>&copy; 2011</p> 

</div><!-- 页 脚 --> 

8 

9 </div><!-- 完成 页 面 --> 

10 </body> 

11 </html><?php // 脚本 8-12 - footer.html #2 

12 


13 // 将 缓冲 内 容 发 送 到 浏览 器 并 关闭 输出 缓冲 : 
14 ob end flush(); 
15 ?> 


该 代码 将 积累 下 来 的 缓 州 
都 是 在 此 刻 发 送 的 。 

(5) 保存 这 两 个 文件 ， 并 将 其 放置 在 启用 了 PHP 的 服务 器 上 的 templates 目 录 中 。 

(6) 在 Web 浏 览 器 中 测试 任意 一 个 页 面 (参见 图 8-31)。 


如 加 加 凶 


已 





内 容 发 送 到 Web 浏 览 器 , 并 关闭 输出 缓冲 。 换 名 话说 , 所 有 的 HTML 
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Raise High the Roof Beam, Carpenters and Seymour: An Introduction 


9:25 pm Monday December 13 


Six Shooter Media 











图 8-31 ”站 点 一 如 既往 地 工作 ,但 当 在 使 用 本 章 稍 后 介绍 的 HTTP 头 之 后 ， 
将 变 得 更 加 容易 
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ww 提示 

口 要 澄清 的 是 ， 如 果 该 文件 会 被 包含 到 PHP 脚 本 (如 index.php) 中 的 话 (就 像 这 里 这 两 
个 示例 一 样 ) ，PHP 代 码 可 以 放置 在 有 .phtml 扩展 名 的 文件 中 。 

口 现在 ，PHP 的 默认 配置 会 自动 启用 输出 缓冲 。 

口 可 以 在 php .ini 文件 中 设置 最 大 缓冲 大 小 ， 默 认 值 是 4 096 字 节 。 

口 ob_get_length () 国 数 返回 当前 缓冲 内 容 的 长 度 (字符 数 )。 

口 ob_get_contents () 函数 可 以 返回 当前 缓冲 区 的 内 容 , 如 果 需 要 的 话 可 以 将 其 赋值 给 一 


= 
个 变量 。 


D ob_flush() 函数 可 以 将 缓冲 中 的 当前 内 容 发 送 到 Web 浏 览 器 中 ， 并 丢弃 这 些 内 容 ， 以 便 
能 够 启动 新 的 缓冲 。 该 国 数 可 以 使 脚本 保持 更 稳定 的 缓冲 大 小 。 

口 ob_clean () 函数 删除 缓冲 中 的 当前 内 容 ， 而 不 会 停止 缓冲 过 程 。 

口 如 果 没 有 调用 ob_end_flush() 函数 , PHP 会 在 脚本 结束 后 自动 调用 它 。 但 自己 调用 该 函 
数 仍然 是 一 种 好 习惯 。 


8.9 处 理 HTTP 头 


服务 器 和 Web 训 览 器 (客户 端 ) 的 很 多 交互 都 是 通过 HTTP (Hypertext Transfer Protocol) 进 
行 的 。 这 就 是 Web 页 面 的 地 址 都 要 以 http:// 开 头 的 原因 。 但 是 ， 除 了 发 送 HTML、 图 片 等 信息 外 ， 
Web 服 务 器 通常 需要 用 其 他 一 些 方 式 与 客户 端 通信 。 这 些 额 外 的 通信 和 需要 使 用 HTTPD 实现 。 
HTTP 头 有 很 多 种 用 法 ， 所 有 这 些 HTTP 头 都 可 以 使 用 PHP 的 header () 函数 来 处 理 。 

本 市 演示 了 header () 函数 的 一 种 常见 的 用 法 : 将 用 户 从 一 个 页 面 重 定向 到 另 一 个 页 面 。 要 
使 用 PHP 来 重 定 向 用 户 的 浏览 器 ， 需 要 发 送 一 个 Ilocation 头 ， 

header ('Location: page.php'); 
通常 , 该 header () 函数 会 后 跟 一 个 exit () ， 用 以 取消 当前 脚本 的 执行 (因为 浏览 器 已 经 被 
重 定向 到 另外 一 个 页 面 了 ): 


header ('Location: page.php'); 
exit(); 


使 用 header () 需 要 了 解 的 最 重要 的 一 件 事 就 是 ， 必 须 在 任何 内 容 被 发 送 到 Web 浏 览 器 之 前 
调用 该 函数 一 一 否则 ， 会 看 到 非常 常见 的 “HTTP 头 已 发 送 ” 错 误 消 息 (参见 图 8-30)。 如 果 Web 
浏览 器 接收 到 了 任何 HTML 或 哪怕 是 一 个 空格 ，header () 函数 都 无 法 工作 。 

幸运 的 是 ，8.8 市 已 经 介绍 了 输出 缓冲 。 因 为 在 Web 应 用 程序 中 打开 了 输出 缓冲 ， 所 以 任何 内 
容 都 不 会 立即 发 送 给 Web 浏 览 器 ， 直 到 页 脚 脚 本 的 最 后 一 行 〈 当 调用 ob_enq_flush() 时 )。 通 
过 这 种 方法 ， 可 以 避免 “HTTP 头 已 发 送 ”的 错误 。 

为 了 演示 重 定向 ， 下 面 我 们 重 写 登录 页 ， 在 成 功 登 录 后 将 用 户 转 到 欢迎 页 。 
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这 使 用 header() 函数 








(1) 在 文本 编辑 器 或 IDE 中 打开 login.php (参看 脚本 8-8)。 
(2) 删除 “You are logged in...” 这 条 print 语 句 (参看 脚本 8-13)。 


脚本 8-13 ”新 版 本 的 登录 页 会 使 用 header () 函数 将 用 户 重 定向 到 另 一 个 页 面 


‘OOo~OUORODPp 


o own wb PR 





33 





<?php // 脚本 8-13 - login.php #2 
/* 页 面 可 以 用 于 用 户 登 录 (理论 上 )。*/ 


// 设置 页 面 标题 并 包含 页 头 文 件 : 
define('TITLE', 'Login'); 
include('templates/header.html'); 


// 打印 一 些 介绍 文本 : 

print '<h2>Login Form</h2> 
<p>Users who are logged in can take advantage of certain features like this, that, 
and the other thing.</p>'; 


// 检查 表单 是 否 已 提交 : 





if ($_SERVERI['REQUEST METHOD'] == 'POST') { 
// 处 理 表单 : 
If ( (!empty($_POST['email'])) && (!empty($_POST['password'])) ) { 
if ( (strtolower($_ POST['email']) == 'me@example.com') && ($_POST['password'] 
== 'testpass') ) { // 相等 ! 


// 将 用 户 重 定向 到 其 他 页 面 ! 
ob_enqd clean(); // 清空 绥 冲 ! 
header ('Location: 

welcome .php'); 

exit(); 





} else { // 不 相等 ! 


print '<p>The submitted email address and password do not match those on 
file!<br/>Go back and try again.</p>'; 


} 
} else { // 表单 未 填 完整 。 


print '<p>Please make sure you enter both an email address and a password!<br 
/>Go back and try again.</p>'; 


} 
) else { // 显示 表单 。 
print '<form action="login.php" method="post"> 


<p>Email Address: <input type="text" name="email" size="20" /></p> 
<p>Password: <input type="password" name="password" size="20" /></p> 
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42 
43 
44 
45 
46 
47 
48 


<p><input type="submit" name="submit" Value="Log In!" /></p> 
</form>'; 

} 

include('templates/footer.html'); // 包含 页 脚 文件 。 

?> 


由 于 用 户 将 被 重 定向 到 其 他 页 面 ， 因 此 这 里 不 再 需要 这 条 消息 了 。 
(3) 在 原来 print 语 句 的 地 方 ， 添 加 : 


ob_end clean( ); 
header ('Location: welcome.php'); 
exit( ); 


第 一 行 代码 销毁 了 页 面 缓冲 (因为 此 时 积累 下 来 的 缓冲 不 再 有 用 )。 这 样 做 并 不 是 必需 的 ， 








但 的 确 是 个 好 主意 。 下 一 行 代码 将 用 户 重 定向 到 welcome .php。 第 三 行 代码 结束 了 该 脚本 剩余 
部 分 的 执行 。 
(4) 保存 该 文件 ， 并 放 到 启用 了 PHP 的 服务 器 上 的 适当 目录 中 (和 本 章 其 他 脚本 放 在 一 起 )。 
接 下 来 需要 创建 一 个 welcome .php 页 面 ， 用 户 将 被 重 定向 到 该 页 面 。 


今 编 写 weLcome .php 











(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 PHP 文 档 ， 命 名 为 velcome .php (参看 脚本 8-14): 


<?php // 脚本 8-14 - welcome.php 





脚本 8-14 ”欢迎 页 会 在 用 户 登录 之 后 显示 欢迎 辞 


OJINAODP 
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心 
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16 


<?php // 脚本 8-14 - welcome.php 
/* 这 是 欢迎 页 ， 用 户 成 功 登 录 后 
会 被 重 定向 到 这 里 。*/ 


// 设置 页 面 标题 并 包含 头 文件 : 
define('TITLE', 'Welcome to the J.D. Salinger Fan Club!'); 
include('templates/header.html'); 





// 关闭 PHP 节 创建 HTML 内 容 : 


?> 


<h2>Welcome to the J.D. Salinger Fan Club!</h2> 

<p>You've successfully logged in and can now take advantage of everything the site 
has to offer.</p> 

<p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor 
incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud 
exercitation ullamco laboris nisi ut alidquip ex ea commodo consequat. Duis aute 
irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla 
pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia 
deserunt mollit anim id est laborum.</p> 


<?php include('templates/footer.html'); // 包含 页 脚 文 件 。?> 


(2) 定义 页 面 的 标题 ， 并 包含 页 头 文件 : 
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define('TITLE', 'Welcome to the J.D. Salinger Fan Club!'); 
include('templates/header.html'); 


(3) 创建 页 面 内 容 : 


? 

<h2>Welcome to the J.D. Salinger Fan Club!</h2> 

<p>You've successfully logged in and can now take advantage 
一 of everything the site has to offer.</p> 


(4) 返回 PHP 并 包含 页 脚 文件 : 
<?php include('templates/footer.html'); ?> 


(5) 将 脚本 保存 为 velcome .php， 放 置 到 与 新 版 本 的 login .php 相 同 的 目录 中 ， 并 在 Web 浏 
览 器 中 进行 测试 (参见 图 8-32、 图 8-33 和 图 8-34)。 
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图 8-32 ”登录 表单 图 8-33 ”以 及 当 用 户 成 功 登录 后 进行 的 重 定向 








Logm Form 


Users who are logged in can take advantage of certain features like this, 


that, and the other thing 


The 


submitted email address and password do not match those on file! 








Co back and try again 








图 8-34 ”如 果 用 户 没 有 登录 成 功 ， 则 继续 留 在 登录 页 





ww 提示 

口 如 果 训 览 器 已 经 收 到 HTTP 头 ， 则 headqers_sent () 国 数 会 返回 TRUE， 此 时 headqer () 国 
数 将 不 能 使 用 。 

口 下 面 的 代码 利用 GET 方 法 的 方式 ， 将 值 从 一 个 页 面 传递 到 另 一 个 页 面 : 
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Svar = urlencode('Pass this text'); 
header ("Location: page.php? message=$var"); 


口 在 重 定向 时 ，heaqer () 函数 从 技术 上 讲 应 该 对 目标 页 使 用 完整 路 径 。 例 如 ， 应 该 是 : 
header ('Location: http://www.example.com/welcome.php'); 
或 
header ('Location: http://localhost/welcome.php'); 


口 在 我 的 《PHP 6 与 MySQL 5 基础 教程 》 一 书 中 ， 我 给 出 了 一 些 代 码 ， 能 够 基于 当前 脚本 的 
位 置 动态 地 生成 一 个 绝对 URL。 


8.10 ”回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
com/forum/。 




















8.10.1 回顾 


口 include() 与 required() 的 区 别 是 什么 ? 

口 为 什么 可 以 将 PHP 代 码 放 在 扩展 名 是 .html 的 包含 文件 中 ? 

口 相对 引用 一 个 文件 与 绝对 引用 一 个 文件 的 区 别 是 什么 ? 

口 如 何 定义 一 个 常量 ? 常量 名 区 分 大 小 写 吗 ? 如 何 检查 常量 是 否 已 经 定义 ? 
口 epoch 是 什么 ?时 间 截 是 什么 ? 

口 以 下 代码 是 什么 意思 : $_SERVER['REOUEST_METHOD']9? 

口 如 何 让 表单 元 素 “ 记 住 ”之 前 提交 过 的 值 ? 

口 如 何 看 到 表单 元 素 内 出 现 的 PHP 错 误 信息 (如 显示 表单 元 素 的 值 时 ) ? 

口 “HTTP 头 已 发 送 ”(headers already sent) 错误 是 什么 意思 ?如何 避免 ? 














8.10.2 ”实践 


口 为 本 章 的 示例 创建 一 个 新 的 设计 原型 , 之 后 创建 新 的 header 和 footer 文 件 。 再 次 浏览 站 
点 的 任意 页 面 (不 需要 改变 任何 PHP 脚 本 ) 。 

D 在 header .html 中 修改 date() 国 数 的 参数 ， 以 不 同 的 方式 显示 日 期 和 (或 ) 时 间 。 

口 重 写 register .php 中 有 关 密 码 的 条 件 语 句 ， 使 用 一 对 骨 套 的 条 件 语句 。 提 示 : 参看 第 6 
章 的 示例 。 

口 如 果 使 用 的 是 5.2 及 以 上 版 本 的 PHP, 查找 PHP 手 册 中 的 Filter extension。 在 register .php 
中 借助 filter_var () 函数 验证 Email 地 址 。 

口 修改 发 送 注 册 信 息 的 Email 的 主题 和 内 容 ， 使 其 更 生动 ， 信 息 更 完整 。 











第 9 章 
cookie 和 session 








本 章 内 容 

口 什么 是 cookie 

口 创建 cookie 

口 读 取 cookie 

口 向 cookie 添 加 参数 
口 删除 cookie 

口 创建 Session 

口 访问 session 变 量 
口 删除 session 

口 回顾 和 实践 


第 8 章 讲 述 了 一 些 较为 复杂 的 Web 应 用 程序 开发 技术 。 当 开始 组 建 一 个 多 页 面 的 Web 站 点 时 很 
可 能 遇 到 的 问题 之 一 是 HITP， 它 是 一 种 无 状态 技术 。 这 意味 着 作为 一 个 Web 开 发 者 没有 内 置 的 
方法 用 以 跟踪 一 个 用 户 或 者 记录 应 用 程序 中 的 某 个 页 面 传递 到 下 一 个 页 面 的 数据 。 这 是 很 严重 的 
问题 ， 因 为 电子 商务 应 用 程序 、 用 户 注册 和 登录 系统 ,以 及 其 他 常用 的 在 线 服务 都 依赖 于 这 个 功 
能 。 幸 运 的 是 ， 使 用 PHP 维 护 从 一 个 页 面 到 另外 一 个 页 面 的 状态 是 非常 容易 的 。 

本 章 将 讨论 两 种 用 以 跟踪 数据 的 主要 方法 : cookie 和 session。 从 如 何 创建 、 读 取 、 修 改 和 删 
除 cookie 入 手 ， 然 后 轻松 掌握 session， 更 加 有 效 地 维护 状态 。 








9.1 什么 是 cookie 


在 cookie 出 现 之 前 ,浏览 Web 网 站 是 种 没有 历史 可 言 的 “旅程 。 虽 然 浏览 器 会 跟踪 所 访问 的 
页 面 ， 允 许 使 用 后 退 〈Back) 按钮 返回 到 之 前 访问 过 的 页 面 ， 并 且 使 用 不 同 颜色 标注 出 已 经 访问 
过 的 链接 ， 但 是 服务 器 并 不 会 记录 谁 访问 过 什么 内 容 。 如 果 站 点 不 使 用 cookie， 或 者 用 户 在 Web 
浏览 器 中 禁用 了 cookie， 那 么 服务 器 端 也 不 会 记录 任何 内 容 (参见 图 9-1)。 

为 什么 这 会 成 为 一 个 问题 ? 如 果 服 务 器 不 能 跟踪 用 户 , 将 不 能 使 用 购物 车 进行 在 线 采 购 。 如 
果 cookie 不 存在 (或 者 如 果 它 们 在 Web 浏 览 器 中 被 禁用 了 )， 人 们 将 不 能 使 用 要 求 用 户 注册 的 那些 
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流行 的 站 点 。 简 言 之 ,没有 cookie 就 没有 Amazon 和 Facebook 或 其 他 流行 、 有 益 的 站 点 (或 至 少 不 
是 现在 的 样子 )。 


| 四 


@ You can choose how cookies are handled in the Internet 
漠 zone. This overrides automatic cookie handling. 


Cookies 


Dvernde automatic cookie handling 


First-party Cookies Third-party Cookies 
© accept ©) Accept 

© Block © Block 

OO Prompt OO Prompt 


[Always allow session cookies 








| 





9-1 大 多 数 Web 训 览 器 让 用 户 能 够 对 处 理 cookie 的 参数 进行 设置 。 这 是 

Internet Explorer 8 高 级 隐私 设置 (Advanced Privacy Settings) 标签 

cookie 是 服务 器 在 用 户 计算 机 上 保存 用 户 信 息 的 一 种 方式 ， 以 便服 务 器 能 够 在 访问 过 程 中 或 
者 多 次 访问 中 记 住 用 户 。cookie 就 像 一 个 名 称 标签 ， 用 户 计算 机 告知 服务 器 用 户 名 称 ， 并 且 给 予 
一 个 名 称 标签 。 然 后 服务 器 能 够 通过 名 称 标签 获知 用 户 是 谁 。 

这 引出 cookie 另 外 一 个 有 关 安 全 的 问题 。cookie 有 不 好 的 一 面 ， 这 是 因为 人 们 认为 它 让 服务 
器 获取 了 过 多 关于 用 户 的 信息 。 但 是 ，cookie 只 存储 传递 给 它 的 信息 ， 因 此 它 可 以 达到 期 望 的 安 
全 程度 。 如 前 所 述 ， 现 代 浏 览 器 都 可 以 根据 需要 定制 cookie 处 理 。 

PHP 对 cookie 提 供 很 好 的 支持 。 本 章 将 教授 如 何 建立 cookie， 从 cookie 中 进行 信息 检索 ， 以 及 9 
如 何 删除 cookie。 还 将 看 到 一 些 可 以 用 来 限制 cookie 的 可 选 参数 。 

在 往 下 进行 之 前 ， 需 要 对 cookie 了 解 两 点 。 首 先是 如 何 调试 与 cookie 相 关 的 问题 。 这 将 在 杠 
注 中 进行 讨论 。 第 二 点 是 如 何 传输 和 接收 cookie (参见 图 9-2)。cookie 存 储 在 Web 浏 览 器 中 ,但 是 
只 有 最 初 发 送 cookie 的 站 点 才能 够 读 取 它 。 同时 , cookie 在 Web 浏 览 器 对 站 点 中 的 页 面 发 出 请 求 时 
被 站 点 读 取 。 换 句 话 说， 当 用 户 在 地 址 栏 中 输入 URL 地 址 并 点 击 转 到 (GO 或 者 类 似 的 其 他 按钮 ) 
时 ， 站 点 就 会 读 取 所 有 它 能 访问 的 cookie 并 且 处 理 所 请 求 的 页 面 。 这 个 顺序 非常 重要 ， 这 是 因为 
它 指 明了 访 癌 cookie 的 时 间 和 方式 。 

w 提示 

口 服务 器 端的 PHP 和 浏览 器 端的 JavaScript 都 能 发 送 、 读 取 和 删除 cookie， 这 是 这 两 种 脚本 的 

重合 功 能 之 一 。 
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Web 浏 览 器 《客户 端 ) 服务 器 
己 -二 Il 了 | 
本 
6 
LURL 请 求 
2. cookies, and HTML 





PHP 脚 本 


3. URL 请 求 and cookic、 1 


HTML 页 面 





PHP 脚 本 























图 9-2 ”在 客户 端 /服务 器 之 间 来 回 传 递 cookie 


调试 cookie 

UPHPUDUUUDUcookeUUUUOOOODOUOOOODODOUO0O0OODODOU0O0O00 
U000 
DOD PHPDN UU cookiel] 
DD Webl 0 0DUDD cookied 
D0 PHPL OU UU cookied 

国生 
UDOUOUOUOOUOOODOUOUwUDOUYwesbuDOO0OD0UcookeuU0OOoOouoo000 Webd 0 
U0O00cookeUUDO0OD0OD 

DD Windows[| Internet Explorer[| D DUODDOODOODOODOODOODOD ternetDUDDODODOODOODOODODODO 
DO0000000000000“0000cookid0 DO000%D00000000000 
Dcookie] O00UUUO0UUUUUUU cookiell PternetExplorerlUUUOOOOO0OO0ODO 
DOUUUUOUU Iternet ExplorerlUUUDOOUODODOUO0O0D0DD Developer Toolsd (00 
JoOogoggoggdn 

DOD00UUU Firefoxd [DU cookie] O00OUUUDcookied D0UU [WD extensions[ 
DD Firecookie[l] FirebusUUUUOOOOOUDOUUOODOUDUcooke0uuouUoouooo0 
DDD Privacyd D0DDU OptionsU Preferences[| [| D0 U0 Use custom settings for history[| [| 
DOD0OUO0O0UUUcookied 0000U0000000U000 

UL Mac OS X[D Windows[| [DO 0 Safari] DOD0UUUUDU cookied DUUUU Preferences 
UUUUsecunatyUUUUOOOOO0O000 

D000000000000U00Ucookiead O00000000000000UU000 
中 加 gg 
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9.2 创建 cookie 





理解 cookie 的 一 个 重要 事情 是 ， 它 们 必须 在 发 送 其 他 任何 信息 之 前 从 服务 器 发 送 到 客户 端 。 
也 就 是 说 ， 脚 本 应 该 在 print 语 句 之 前 ,或 引入 任何 包含 HTML 的 外 部 文件 之 前 发 送 cookie。 

如 果 服 务 器 尝试 在 Web 浏 览 器 已 经 获得 了 HTML 代 码 之 后 发 送 cookie， 甚 至 是 无 关 紧 要 的 空 
格 之 后 ， 都 将 产生 一 个 错误 消息 并 且 cookie 也 将 发 送 失败 (参见 图 9-3)。 这 是 迄今 为 止 最 常见 的 
与 cookie 相 关 的 错误 。 


加 国 E 








Warning: Cannot modify header nformation - headers already sent by (output 
statted at /Usersilarryullman Sites/phpvdqsdicustomize.php:2) in /Users 
larryullmanmSites/phpvqsdicustomize.php on line 8 


Warning: Cannot modify header information - headers already sent by (output 
statted at /Usersilarryullman Sites/phpvdqsdicustormize.php:2) n /Users 
/arryullmanm’Sitesiphpvqs4/customize.php on line 9 


Your settings have been entered! Click here to see them tn action. 





Use this form to set your preferences: 


Font Size Y||FontColor Y Set My Preferences 











图 9-3 ”如 果 函 数 setcookie() 在 任何 信息 被 发 送 至 浏览 器 之 后 被 调用 ， 
其 至 发 送 的 是 一 个 空白 行 ， 将 看 到 一 个 类 似 这 样 的 消息 


使 用 函数 setcookie() 发 送 cookie: 


setcookie (name, value); 
setcookie('CookieName', 'This is the cookie value.'); 


这 行 代 码 向 浏览 器 发 送 了 一 个 名 为 CookieName 的 cookie， 其 值 为 This is the cookie 
value (参见 图 9-4)。 

尽管 向 相同 站 点 发 送 cookie 的 数量 已 经 被 Web 浏 览 器 所 限制 ， 不 过 还 是 可 以 用 函数 
setcookie() 继续 向 浏览 器 发 送 更 多 的 cookie。 


setcookie('name2', 'some value'); 
setcookie('name3', 'another value'); 


最 后 ， 正 如 即将 在 本 示例 中 看 到 的 那样 ， 当 创建 cookie 时 ， 可 以 使 用 一 个 变量 作为 cookie 的 
名 称 或 者 值 属性 。 


Setcookie(Scookie_name，S$cookie_value) ; 


作为 设置 cookie 的 示例 ， 这 里 将 创建 一 个 脚本 以 让 用 户 来 指定 页 面 文本 的 字体 大 小 和 颜色 。 
页 面 将 显示 在 表单 中 选择 这 些 值 的 结果 (参见 图 9-5) ， 然 后 提交 表单 〈 参 见 图 9-6) 。 本 音 的 9.3 市 
将 创建 出 一 个 单独 的 页 面 ， 它 将 用 到 这 些 设置 。 
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@ -The website "10.0.1.2" has requested to save a file on your 
二 ;computer called a "cookie.” This file may be used to track usage 
information. Do you want to allow this? 


[Apply my decision to all cookies from this website 


[aow Cookie | [Biock Cookie | More Info 


Cookie Information 











Name CookieName 

Doman |10012 

Path 六 

Expires Em 了 Secure No | 
Data This+is+the+cookie+walue. 

3rdParty No Session Yes | 

Compact l 本 

Policy 











图 9-4 如果 浏 览 器 被 设置 为 提示 用 户 使 用 了 cookie， 在 每 次 发 送 cookie 时 将 会 出 现 一 
个 这 样 的 消息 (注意 Internet Explorer 8 的 这 个 窗口 显示 了 URL 编 码 格式 中 的 值 ) 





Use this fotrm to set your preferences: 


Font Size | Font Color Y | Set hly Preferences 


图 9-5 ”该 表单 用 于 选择 另外 一 个 PHP 页 面 使 用 的 字体 大 小 和 颜色 

















Your settings have been entered! Click here to see them in action. 





Use this form to set your preferences: 


Font Size ™ |FontColor Y% | Set My Preferences 


图 9-6 ”在 提交 表单 之 后 ， 页 面 显示 出 一 条 消息 和 指向 另外 一 个 页 面 
(将 使 用 用 户 选 择 的 页 面 ) 的 链接 。 接 下 来 将 创建 这 个 页 面 



































瀛 发 送 cookie 








(GD 在 文本 编辑 器 或 者 IDE 中 创建 一 个 新 的 PHP 文 档 , 命 名 为 customize.php( 参 看 脚本 9-1): 


<?php // 脚本 9-1 - customize.php 
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脚本 9-1 两 个 cookie 将 用 来 存储 用 户 对 于 文本 的 字体 大 小 和 颜色 的 选择 。 该 页 面 将 显示 并 处 理 
这 个 表单 








1 <?php // 脚本 9-1 - customize.php 
2 
3  ”// 如 果 表 单 已 提交 ， 则 处 理 它 : 
4 if (isset($_POST['font_ size'], $_POST['font color'])) { 
5 
6 // 发 送 cookie: 
7 setcookie('font size', $_ POST['font size']); 
8 setcookie('font color', $_ POST['font color']); 
9 
10 // 打印 一 条 消息 : 
Smsg = '<p>Your settings have been entered! Click <a href="view settings.php"> 
here</a> to see them in action.</p>'; 
2 
3 】 // 结束 条 件 语 向 。 
14 ?><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
15 "http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
6 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 
7 <head> 
18 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
19 <title>Customize Your Settings </title> 
20 </head> 
21 <body> 


22 <?php // 如 果 cookie 已 发 送 ， 打 印 一 条 消息 。 
23 if (isset(Smsg)) { 








24 print S$msg; 

25 } 

26 ?> 

27 

28 <p>Use this form to set your preferences:</p> 
29 

30 <form action="customize.php" method="post"> 
31 <select name="font_size"> 

3 了 2 <option value="">Font Size</option> 

33 <option value="xx-small">xx-small </option> 
34 <option value="x-small">x-small </option> 
35 <option value="small">small </option> 

36 <option value="medium">medium </option> 

3 <option value="large">large </option> 

38 <option value="x-large">x-large </option> 
39 <option value="xx-large">xx-large </option> 
40 </select> 

41 <select name="font_ color"> 

42 <option value="">Font Color </option> 

43 <option value="999">Gray</option> 

44 <option value="0c0">Green</option> 

45 <option value="00f">Blue</option> 

46 <option value="c00">Red</option> 

47 <option value="000">Black</option> 

48 </select> 

49 <input type="submit" name="submit" value="Set My Preferences" /> 
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50 </form> 


52 </body> 
53” "</html> 


cookie 最 重要 的 问题 是 它们 在 向 Web 浏 览 器 发 送 任何 信息 之 前 被 创建 。 为 了 实现 这 个 目的 ， 


这 些 脚 本 位 于 处 理 cookie 发 送 的 PHP 代 码 之 前 。 





同时 请 确认 PHP 起 始 标签 之 前 不 要 有 空格 或 空 行 。 
(2) 检验 表单 是 否 被 提交 
if (isset($_POST['font_ size'], $_POST['font color'])) { 


该 页 面 会 显示 并 处 理 表 单 。 它 会 使 用 第 8 章 介绍 的 方法 ( 即 检 查 $_SERVERI['REQUEST_ 


METHOD' ] 变量 的 值 是 否 为 POST) 验证 表单 是 否 提 交 , 但 这 里 采用 的 是 另外 一 种 方式 ,实现 最 基 


本 、 





col 





最 小 限度 的 验证 。 条 件 语句 检查 是 否 已 经 设置 了 $_POST['font_size'] 和 $_POST['font_ 
or '] 变量 。 如 果 两 个 变量 都 已 设置 ， 就 表明 表单 已 提交 。 
(3) 创建 cookie: 


setcookie('font size', $_POST['font size']); 
setcookie('font color', $_POST['font_ color']); 


这 两 行 代 码 创建 了 两 个 单独 的 cookie: 一 个 名 为 font_size， 另 一 个 名 为 font_color。 它 











们 的 值 将 取决 于 HIML 表 单 中 所 选 的 值 ， 这 些 值 被 存 贮 在 变量 $_PosT['font_size'] 和 
S$_POoST['font_ color'] 中 。 


后 在 脚本 中 会 被 使 用 并 用 以 打印 一 条 消息 。 这 一 步 很 重要 ， 因 为 不 能 在 连接 处 (这 时 还 没有 创建 


在 实际 开发 的 应 用 程序 中 , 我 会 在 cookie 使 用 表单 提交 的 元 素 之 前 , 对 用 户 选 择 的 值 进行 验证 。 
(4) 创建 一 条 信息 ， 并 且 完 成 条 件 语句 和 PHP 代 码 段 : 

$msg = '<p>Your settings have been entered! Click <a href= 

"view_ settings.php">here</a> to see them in action.</p>'; 


】// 结束 条 件 语句 。 


学 渤 


当 表单 被 提交 时 ,将 发 送 cookie， 并 且 变 量 $msg 会 被 赋予 一 个 字符 串 类 型 的 值 。 这 个 变量 稍 











HIML 头 ) 打印 信息 。 





(5) 创建 HIML 头 并 且 开 局 bodqy 标 签 : 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Customize Your Settings </title> 
</head> 
<body> 


所 有 的 这 些 代码 都 必须 在 setcookie () 行 之 后 。 这 里 并 没有 夸大 事实 ， 没 有 文本 、HTML 








或 者 空白 能 够 在 调用 setcookie() 之 前 发 送 至 Web 浏 览 器 。 
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(6) 创建 另外 一 个 PHP 代 码 段 以 报告 cookie 已 经 被 发 送 : 


<?php 

if (isset($msg)) { 
print $msg; 

} 


ns 


如 果 cookie 已 经 被 发 送 ， 这 些 代码 将 会 打印 出 一 条 消息 。 用 户 第 一 次 访问 页 面 时 ，cookie 并 
` 会 被 发 送 ， 因 此 不 会 设置 $msg, 这 个 条 件 的 结果 为 FALSE, 并 且 print 调 用 不 会 运行 。 一 旦 提 
交 表 单 ， 就 会 设置 Smsg， 这 个 条 件 的 结果 将 为 TRUE。 

0) 开始 编写 HTML 表 单 : 


<p>Use this form to set your preferences:</p> 
<form action="customize.php" method="post"> 
<select name="font_size"> 
<option value="">Font Size</option> 
<option value="xx-small">xx-small </option> 
<option value="x-small">x-small </option> 
<option value="small">small</option> 
<option value="medium">medium </option> 
<option value="large">large</option> 
<option value="x-large">x-large </option> 
<option value="xx-large">xx-large </option> 
</select> 
HTML 表 单 非常 简单 (参见 图 9-5)。 用 户 可 以 通过 一 个 下 拉 菜 单 来 选择 字体 大 小 。 每 个 值 都 
对 应 CSS 代 码 以 用 来 设置 文档 的 字体 大 小 : 从 xx-smal1l 到 xx-large。 
因为 这 段 脚 本 显示 并 处 理 表单 ， 所 以 表单 的 行为 属性 将 指向 相同 的 这 个 文件 。 
(8) 完成 HIML 表 单 
<select name="font_ color"> 
<option value="">Font Color </option> 
<option value="999">Gray</option> 
<option value="0c0">Green</option> 
<option value="00f">Blue</option> 
<option value="c00">Red</option> 
<option value="000">Black</option> 
</select> 


<input type="submit" name= "submit" value="Set My Preferences" /> 
</form> 


第 二 个 下 拉 菜 单 用 来 选择 字体 颜色 。 菜 单 显 示 文 本 表单 中 的 颜色 , 但 是 其 值 为 HTML 类 型 的 
颜色 值 。 通 常 这 样 的 值 使 用 六 个 字符 外 加 一 个 # 号 来 表示 (如 #00CcC00)， 但 是 CSS 允 许 仅 使 用 3 
个 字符 的 版 本 ， 并 且 将 在 使 用 这 些 值 的 页 面 中 添加 # 号 。 

(9) 完成 HTML 页: 


</body> 
</html> 


(10) 将 文件 保存 为 customize .php， 并且 放置 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 。 
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(11) 确保 已 经 将 Web 浏 览 器 设置 为 为 每 个 cookie 进 行 提 示 。 
为 了 保证 脚本 能 够 运行 ， 设 置 浏 览 器 为 每 个 cookie 提 供 提 示 。 参 见 “ 调 试 cookie” 框 注 。 
(12) 在 Web 浏 览 器 中 运行 脚本 (参见 图 9-7 和 图 9-8)。 





@ — The website "10.0.1.2" has requested to save a file on your 
.computer called a "cookie.” This file may be used to track usage 
information. Do you want to allow this? 











点 pply my decision to all cookies from this website 





Cookie Information 

Name font_size 

Domain 10.0.1.2 

Path 用 

Expires End of session Secure No 
Data *-large 

3rdParty No Session Yes 

Compact 

Policy 








图 9-7” 如 果 用 户 已 经 选择 了 在 接受 cookie 前 进行 提示 ， 那 么 当 setcookie () 第 一 次 
被 调用 时 , 他 们 将 看 到 这 条 消息 。 存 储 名 为 font_size 且 值 为 x-smal1 的 cookie 









































更 上 《2 HTML Css DOM Cookies ~ 用 日 日 昌 
Cookies” Fiter * Default (Accept cookies)™ Tools™ 
Name Yalue Domain Size Path Expires HttpOnly Security 
3 font_color 999 10.0,1,2 136 | Session 
Yalue 
999 
= font_size -large 10.0.1,2 166 | Session 
Yalue 
x-large 
图 9-8 ”Firefox 的 Firebug 扩 展 包 显示 了 浏览 器 接收 到 的 cookie。 第 2 个 cookie 是 由 PHP 











脚本 发 送 的 ， 名 为 font-color 值 为 999， 代 表 字 体 颜 色 为 灰色 





Vv 提示 

口 cookie 是 PHP 中 为 数 不 多 的 几 个 因为 浏览 器 的 不 同 或 操作 系统 的 不 同 而 不 同 的 工具 之 一 。 

应 当 在 尽 可 能 多 的 浏览 器 和 操作 系统 中 测试 基于 cookie 的 应 用 程序 。 

口 如 果 使 用 第 8 章 中 教授 的 输出 缓存 技术 , 那么 就 可 以 将 setcookie () 调用 放置 在 脚本 中 的 
任何 位 置 (这 是 因为 Web 浏 览 器 在 调用 图 数 ob_enq_flush () 之 前 不 会 获得 数据 )。 
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口 cookie 的 总 数 限制 接近 于 4KB。 这 已 经 远 远 超出 大 多 数 应 用 程序 所 需 的 数量 。 
口 可 以 使 用 headers_sent () 函数 来 测试 发 送 cookie 是 否 安全 。 它 将 报告 HTTP 头 是 否 已 经 
发 送 给 Web 浏 览 器 。 





9.3 读 取 cookie 


就 像 表单 数据 被 存储 在 数组 $_PosT 中 (假定 该 表单 使 用 POST 方 法 )，URL 中 传送 给 脚本 的 
值 被 存储 在 数组 $_GET 中 那样 ， 函 数 setcookie() 将 cookie 数 据 存放 在 数组 $_cooKIE 中 。 从 
cookie 中 获得 某 个 值 ， 只 需要 将 cookie 的 名 称 指定 为 该 数组 的 索引 即 可 。 例 如 ， 获 得 在 下 面 脚本 
创建 的 cookie 值 : 


setcookie('user', 'trout'); 
可 以 使 用 变量 $_COOKIE['user']。 

除非 更 改 了 cookie 的 参数 〈 将 在 本 章 稍 后 部 分 看 到 ) ，Web 应 用 程序 中 每 个 页 面 都 可 以 访问 
cookie。 但 是 应 该 知道 ，cookie 在 未 被 发 送 前 不 能 立即 被 脚本 访问 。 这 里 不 能 这 样 做 : 


setcookie('user', 'trout'); 
print $_ COOKIE['user']; // No value. 


其 中 的 原因 是 cookie 被 读 取 和 发 送 的 顺序 问题 (参见 图 9-2)。 
访问 cookie 的 值 非常 简单 , 我 们 编写 一 段 脚 本 来 使 用 在 customize.php 中 用 来 指定 页 面 文本 
大 小 和 颜色 的 偏好 设置 。 该 脚本 将 使 用 CSS 以 达到 这 样 的 效果 。 


信用 PHP 检 索 cookie 数 据 


























(1) 在 文本 编辑 器 或 者 IDE 中 开启 一 个 新 的 PHP 文 档 ， 命 名 为 view_settings .Php (参看 脚 
本 9-2): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>View Your Settings</title> 





脚本 9-2 该 脚本 使 用 在 cookie 中 存储 的 值 设置 CSS 中 的 字体 大 小 和 颜色 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
<html xmlns="http://ww.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>View Your Settings</title> 
<style type="text/css"> 
body { 
<?php // 脚本 9-2 - view_settings.php 











PO 
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11 // 检查 font_size 的 值 : 
12 if (isset($ COOKIE['font size'])) { 


13 print "\t\tfont-size: " . htmlentities($ COOKIE['font size']) . ";\n"; 
14 } else { 

15 print "\t\tfont-size: medium;"; 

16 } 

于 了 


18 // 检查 font_color 的 值 : 

19 if (isset($ COOKIE['font color'])) { 

20 print "\t\tcolor: #" . htmlentities($ COOKIE['font color']) . ";\n"; 
21 } else { 

22 print "\t\tcolor: #000;"; 


23 } 

24 

25. 有 > 

26 

人 7 </style> 
28 </head> 

29 <body> 


30 <p><a href="customize.php">Customize Your Settings</a></p> 
31 <p><a href="reset.php">Reset Your Settings</a></p> 
32 

33 <p>yadda yadda yadda yadda yadda 

34 yadda yadda yadda yadda yadda 

35 yadda yadda yadda yadda yadda 

36 yadda yadda yadda yadda yadda 

37 yadda yadda yadda yadda yadda</p> 

38 

39 </body> 

40 </html> 


(2) 开始 CSS 代 码 段 : 


<style type="text/css"> 
body { 


这 个 页 面 将 使 用 CSS 来 实现 用 户 的 偏好 设置 。 那 么 要 完成 的 脚本 将 类 似 : 


body { 
font-size: x-large; 
color: #999; 

} 


字体 大 小 和 颜色 的 值 将 根据 用 户 在 customize .php 页 面 中 的 选择 而 不 同 。 在 这 一 步 中 ， 创 
建 了 初始 的 CSS 代 码 。 

(3) 开启 PHP 代 码 段 : 

<?php // 脚本 9-2 - view_settings.php 

该 脚本 将 基于 cookie 用 PHP 来 打印 剩余 的 CSS。 

(4) 如 果 cookie 存 在 的 话 ， 使 用 cookie: 

if (isset(S$_COOKIE['font_size'])) { 


print "\t\tfont-size: " . htmlentities($_ COOKIE['font_size']) . ";\n"; 
} else { 
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print "\t\tfont-size: medium;"; 


} 

如 果 脚 本 能 够 访问 包含 font_size 设 定 的 cookie， 它 将 打印 出 该 cookie 的 值 以 作为 CSS 字 体 
大 小 的 值 。 使 用 isset () 国 数 是 很 必要 的 ， 可 以 用 来 查看 cookie 是 否 存在 。 如 果 不 存在 这 样 的 
cookie，PHP 将 以 默认 大 小 (medium) 打印 。 

出 于 安全 考虑 ，cookie 的 值 不 会 被 直接 打印 出 来 。 取 而 代 之 的 做 法 是 , 运行 通过 在 第 5 章 中 所 
讨论 的 函数 htmlentities() 来 阻止 因为 用 户 对 cookie 的 值 进行 操纵 (这样 做 非常 容易 ) 而 可 能 
造成 的 不 良 后 果 。 

注意 ， 每 个 print 语 句 都 将 创建 两 个 制 表 符 (\t) 和 一 个 换行 符 (\n)。 这 样 做 的 目的 是 让 
将 在 HIML 源 代码 中 出 现 的 CSS 代 码 以 某 种 格式 显示 出 来 .如 果 不 这 样 做 也 不 会 影响 页 面 的 功能 ， 












































(5) 为 保存 字体 颜色 的 cookie 重 复 这 个 过 程 ， 


if (isset($_ COOKIE['font color'])) { 

print "\t\tcolor: #" . htmlentities($ COOKIE['font color']) . ";\n"; 
} else { 

print "\t\tcolor: #000; " 


这 里 的 CSS 颜 色 属 性 已 经 被 赋予 了 一 个 值 。 在 第 4 步 中 cookie 本 身 也 被 相同 的 方式 使 用 。 
(6) 结束 PHP 代 码 段 ， 完 成 CSS 代 码 并 结束 HTML 的 pead 部 分 : 


| } 
</style> 
</head> 
(7) 开始 编写 HTML 的 body 部 分 ， 并 为 两 个 页 面 创 建 链 接 : 
<body> 


<p><a href="customize.php">Customize Your Settings</a></p> 
<p><a href="reset.php">Reset Your Settings</a></p> 


这 两 个 链接 将 用 户 引 向 另外 的 两 个 PHP 页 面 。 第 一 个 是 已 经 编写 出 来 的 customize.php， 
它 用 来 让 用 户 自 定 义 设置 。 第 二 个 页 面 是 reset .php, 本 章 稍 后 将 编写 它 , 这 个 页 面 允 许 用 户 对 
他 们 自 定 义 的 设置 进行 删除 。 

(8) 添加 一 些 文本 : 


<p>yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda</p> 


该 文本 用 来 显示 cookie 变 更 带 来 的 效果 。 
(9) 完成 HTML 页 面 : 


</body> 
</html> 
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(10) 将 文件 保存 为 view_settings .php, 放置 在 同 customize .php 相 同 的 目录 中 , 然后 单 
击 customize.php 上 的 链接 ， 在 Web 浏 览 器 中 测试 (参见 图 9-9)。 





_ View Your Settings - Mozilla Firefox 


Customize Your Settings 


Reset Your Settings 





yadda yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda yadda 
yadda 





图 9-9 ”该 页 面 反应 了 用 PHP 脚 本 所 作 的 自 定义 字体 的 选择 
(11) 查阅 页 面 的 源 代码 以 查看 最 终 的 CSS 代 码 (参见 图 9-10)。 





Source of: http://10.0.1.2:8888/view_settings.ph... [~ |[ 吕 |[X| 
Eile Edt Yiew Help 


<!DOCTYPE htm PUBLIC "~-//W3C//DTID XHTML 1.0 Trans 
"httpi/ /WwW. WI3. Org/ TR/ Khtri i/DID/xhtmi-tra 
<html xmlIns="http:// Wy.w3 .ordg/1999/xhtml" xml:lang 
<head> 
<meta http-equiv="content-type" content="te 
<title>View Your Settings</title> 
<style type="text/css"> 
body { 
font-size: x-large; 
color: #999; 
} 
</style> 
</head> 
<hody> 
<p><a href="customize.php">Customize Your Settings< 
<p><a href="reset .php">Reset Your Settings</a></p> 


<p>vyadda vadda yadda vadda yadda 
yadda vyadda yadda vyadda yadda 
yadda vadda vadda vyadda yadda 
yadda vyadda vadda vyadda vyadda 
yadda vadda vadda yadda yadda</p> 


</hody> 
</html> 


< 





图 9-10 ”通过 查阅 页 面 的 源 代码 ， 还 可 以 对 CSS 值 的 变更 进行 跟踪 
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(12) 使 用 自 定 义 页 面 来 更 改 设置 并 且 返 回 给 这 段 脚 本 。 
每 次 提交 表单 都 将 创建 两 个 用 来 存储 表单 值 的 新 的 cookie， 从 而 替换 已 经 存在 的 cookie。 


Vv 提示 
D cookie 的 值 在 它 被 发 送 时 将 被 自动 编码 ， 并 且 在 被 PHP 页 面 获得 的 时 候 被 自动 解码 。 对 于 
HITML 表 单 发 送 的 值 也 是 如 此 。 


9.4 向 cookie 添加 参数 


虽然 向 国 数 setcookie () 只 传送 name 和 value 参 数 对 于 大 多 数 情况 下 的 cookie 使 用 来 说 已 
经 足够 ,但 是 还 应 当知 道 还 有 其 他 的 可 用 参数 。 函 数 接受 的 参数 能 够 多 达 5 个 ， 每 个 参数 都 用 于 
限制 对 cookie 的 操作 : 

setcookie (name, value, expiration, path, domain, secure, httponly); 

参数 expiration 用 来 为 cookie 的 存在 设 定 一 个 特定 的 时 间 长 度 。 如果 它 没 有 被 指定 , cookie 
将 一 直 起 作用 直到 用 户 关闭 他 们 的 浏览 器 。 通 常情 况 下 , 通过 向 当前 时 间 加 上 一 个 特定 数量 的 分 
钟 或 者 小 时 数 来 指定 过 期 时 间 。 使 用 PHP 的 time () 函数 (参见 第 8 章 ) 获取 当前 时 间 。 下 面 这 行 
代码 将 cookie 的 过 期 时 间 设 定 为 当前 时 间 之 后 的 1 小 时 〈60 秒 乘 以 60 分 钟 ) : 

setcookie (name, value, time()+3600); 

因为 过 期 时 间 将 被 计算 为 time () 的 值 加 上 3600 后 的 结果 ， 这 个 特别 的 参数 没有 被 引号 引用 
(因为 不 希望 逐 字 传送 time () +3600 作 为 过 期 时 间 ， 而 是 传送 计算 结果 )。 

path 和 aqomain 参 数 用 来 限制 在 Web 站 点 (路径 ) 中 的 特定 文件 夹 或 者 特定 域 中 的 cookie。 使 
用 path 选 项 ， 可 以 限制 仅 当 用 户 在 域 中 user 文 件 夹 中 时 cookie 才 存在 : 

setcookie (name, value, time()+3600, '/subfolder/'); 

cookie 已 经 被 指定 给 一 个 域 ， 因 此 域 参数 可 以 用 来 限制 一 个 子 域 的 cookie ， 如 forum. 


example.com, 












































setcookie(name, value, time()+3600, '', 'forum.example.com'); 

参数 secure 的 值 指明 一 个 cookie 应 当 只 能 通过 安全 HTTPS 连 接 传送 。 值 1 指明 必须 使 用 安全 
连接 ， 反 之 值 0 指明 安全 连接 并 不 必要 。 可 以 为 一 个 电子 商务 站 点 确认 一 个 安全 cookie 传 输 : 

setcookie('cart', '82ABC3012', time()+3600, ', 'shop.example.com', 1); 

和 所 有 带 有 参数 的 函数 一 样 ， 必 须 按 顺序 传递 所 有 的 参数 值 。 在 先前 的 示例 中 ， 如 果 不 希 户 
指定 (或 者 限制 ) 路 径 和 域 ， 需 要 使 用 空 的 引号 。 在 path 参 数 中 ， 可 以 使 用 一 个 单独 的 斜 线 (/) 
表示 根 目录 ( 即 ， 无 路 径 重 定向 )。 通 过 这 样 做 ， 可 以 保留 适当 数量 的 参数 并 且 如 果 需 要 仍然 能 
够 指定 使 用 HTTPS 连 接 。 

最 后 一 个 属性 (httponly) 是 PHP5.2 版 本 中 新 添加 的 属性 。 它 可 以 用 来 限制 对 cookie 的 访 
问 例如， 防止 用 JavaScript 读 取 cookie)， 但 不 是 所 有 的 浏览 器 都 对 此 提供 支持 。 
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让 我 们 为 已 经 存在 的 customize .php 页 面 添 加 过 期 日 期 以便 用 户 在 关闭 了 浏览 器 之 后 以 
及 重新 返回 之 前 访问 的 站 点 时 ， 仍 然 保 留 他 们 的 首选 项 。 


过 为 cookie 设 置 过 期 日 期 








(1) 在 文本 编辑 器 或 者 IDE 中 打开 customize.php。( 参 看 脚本 9-1)。 
(2) 按照 下 面 的 代码 修改 2 行 sSetcookie() (参看 脚本 9-3): 


setcookie('font size', $_POST['font size'], time()+10000000, '/', '', 0); 
setcookie('font color', $_POST['font color'], time()+10000000, '/', '', 0); 





脚本 9-3 ”通过 向 这 两 个 cookie 添 加 expiration 参 数 的 方式 , 其 至 可 以 i 上 cookie 持 续 到 用 户 关闭 
和 重新 返回 到 浏览 器 之 后 





1 <?php // 脚本 9-3 - customize.php #2 
2 
3  ”// 如 果 表 单 已 提交 ， 则 处 理 它 : 
4 if (isset($_POST['font_size'], $_POST['font color'])) { 
5 
6 // 发 送 cookie : 
7 setcookie('font size', $_ POST['font size'], time()+10000000, '/'); 
8 setcookie('font color', $_ POST['font color'], time()+10000000, '/'); 
9 
10 // 打印 一 条 消息 : 
二 小 Smsg = '<p>Your settings have been entered! Click <a href="view settings.php"> 
here</a> to see them in action.</p>'; 
这 
3 } // 结束 条 件 语句 。 
14 ?><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
15 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
6 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 
7 <head> 
18 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
19 <title>Customize Your Settings </title> 
20 </head> 
21 <body> 


22 <?php // 如 果 cookie 已 发 送 ， 打 印 一 条 消息 。 
23 if (isset(Smsg)) { 


24 print S$msg; 

25: 了 

26 ?> 

2 

28 <p>Use this form to set your preferences:</p> 
29 

30 <form action="customize.php" method="post"> 
3 <select name="font_size"> 

3 <option value="">Font Size</option> 

33 <option value="xx-small">xx-small </option> 
34 <option value="x-small">x-small </option> 
35 <option value="small">small </option> 

36 <option value="medium">medium </option> 

37 <option value="large">large </option> 

38 <option value="x-large">x-large </option> 


39 <option value="xx-large">xx-large </option> 
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40 </select> 

41 <select name="font_ color"> 

42 <option value="">Font Color</option> 
43 <option value="999">Gray</option> 

44 <option value="0c0">Green</option> 
45 <option value="00f">Blue</option> 

46 <option value="c00">Red</option> 

47 <option value="000">Black</option> 
48 </select> 

49 <input type="submit" name="submit" value="Set My Preferences" /> 
50 </form> 

51 

52 </body> 


53 </html> 


为 了 让 cookie 持 续 一 个 较 长 的 时 间 (持续 数 个 月 ), 可 以 设 定 过 期 时 间 为 从 当前 时 间 开 始 往 后 
10 000 000 秒 。 此 时 ， 设 置 path 参 数 指向 网 站 的 根 目 录 (使 用 /)。 这 样 做 将 可 以 改善 通过 不 同 浏 
览 器 进行 cookie 发 送 时 的 一 致 性 。 

因为 cookie 的 过 期 时 间 被 设置 为 将 来 的 数 月 之 后 ， 保 存在 cookie 中 的 用 户 偏好 设置 在 用 户 关 
闭 以 及 重新 打开 浏览 器 时 仍然 有 效 。 如 果 没 有 过 期 时 间 ， 用 户 将 看 到 默认 的 字体 大 小 和 颜色 ,并 
且 必 须 为 每 个 浏览 器 会 话 重新 分 配 偏好 设置 。 

(3) 保存 文件 ， 放 置 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 , 并 在 Web 浏 览 器 中 重新 测试 ( 参 
见 图 9-11 和 图 9-12 ) 。 




















加 加 加 网 
La ES 了 | 4， Dom Cookies /BD 
Cookies™ Fiter” Default (Accept cookies)* Tools™ 

Name | Yalue | Domain | Size ， Path | Expires HttpOnly | Security 
S font_color 0cn0 10,0,1,2 13B ||} Friday, April 15, , 2011 10:05:47 AM 
Yalue 
oco 
Sfont_size large 10.0,1,2 146 | Friday, April 15, 2011 10:05:47 AM 
Yalue 
large 








图 9-11 如 果 将 浏览 器 设置 为 当 接 受到 cookie 时 提供 提示 ， 将 看 到 过 期 
时 间 是 如 何 被 添加 的 请 同 图 9-6 进 行 对 比 ) 


加 辐 加 四 














Customize Your Settings 


Reset Your Settings 





yadda yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda yadda 
yadda 








图 9-12 ”新 的 cookie 参 数 并 没有 对 应 用 程序 的 功能 起 到 相反 的 作用 
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Vv 提示 

DO 请 注意 ， 当 从 自己 的 计算 机 (如 从 localhost) 发 送 cookie 时 ， 有 的 浏览 器 也 许 不 能 对 

cookie 调 整 后 的 过 期 时 间作 出 反应 。 

口 对 于 在 使 用 cookie 时 选用 何 种 过 期 时 间 来 说 有 几 个 常用 的 原则 。 如 果 cookie 持 续 的 时 间 同 
用 户 浏览 网 站 的 时 间 一 样 长 ,那么 不 要 设置 过 期 时 间 。 如 果 cookie 需 要 在 用 户 关 闭 和 重新 
开启 浏览 器 之 后 继续 存在 ,那么 将 过 期 时 间 设 置 为 数 月 后 的 将 来 。 如 果 cookie 会 造成 一 些 
安全 隐患， 将 过 期 时 间 设 置 为 一 个 小 时 , 或 者 一 小 段 时 间 ， 这 样 cookie 不 会 在 用 户 离开 浏 
览 器 之 后 很 久 依旧 存在 。 

口 当 服务 器 和 客户 端 在 不 同 的 时 区 内 时 ，cookie 的 过 期 时 间 将 会 造成 较 大 的 麻烦 。 过 期 时 间 
是 基于 服务 器 的 时 区 而 设 定 的 , 但 是 浏览 器 可 能 会 删除 基于 客户 端 时 区 的 cookie。 幸 运 的 
是 ， 一些 浏览 器 能 够 自动 对 此 进行 正确 操作 。 

口 出 于 安全 考 虚 ， 可 以 为 cookie 设 定 一 个 5 分 钟 或 10 分 钟 的 过 期 时 间 , 并 且 让 cookie 在 用 户 每 
访问 一 个 新 页 面 时 就 被 重新 发 送 一 次 。 通 过 这 种 方式 , cookie 能 够 在 用 户 访问 的 全 程 存在 ， 
而 在 用 户 离开 后 5 分 钟 或 10 分 钟 后 消失 。 


9.5 删除 cookie 


使 用 cookie 需 要 了 解 的 最 后 一 件 事情 是 如 何 删 除 它 。 虽 然 cookie 会 在 用 户 的 浏览 器 关闭 或 者 
过 期 日 期 /时 间 到 达 时 自动 过 期 ， 但 是 通常 还 是 需要 能 够 手动 删除 它 。 例 如 ， 用 来 进行 用 户 注册 
和 登录 的 Web 站 点 通常 都 能 够 在 用 户 注销 时 删除 所 有 的 cookie。 

函数 setcookie () 接受 7 个 参数 , 但 是 只 有 name 这 个 参数 是 必须 的 。 如 果 传 送 一 个 有 名 称 而 
无 值 的 cookie， 它 起 的 作用 同 删除 一 个 已 经 存在 的 同名 cookie 一 样 。 例 如 ， 使 用 下 面 的 代码 来 创 
建 名 为 username 的 cookie 

setcookie('username', 'Larry'); 

删除 名 为 Lsername 的 cookie， 可 以 编写 代码 


setcookie('username', ''); 






















































































setcookie('username', FALSE); 

为 了 更 加 谨慎 ， 还 可 以 为 之 设 定 一 个 在 过 去 的 过 期 时 间 (参见 图 9-13): 

setcookie('username', FALSE, time() - 600); 

当 删 除 cookie 时 唯一 需要 告诫 的 是 ， 必 须 使 用 同 首 次 设置 cookie 时 相同 的 参数 值 (除了 值 和 
过 期 时 间 )。 例 如 ， 如 果 在 创建 cookie 时 提供 了 domain 值 ， 那 么 在 删除 该 cookie 的 时 候 也 应 该 提 
供 同样 的 值 : 


setcookie('user', 'larry', time() + 3600, '', 'forums.example.com'); 
setcookie('user', '', time() - 600, '', 'forums.example.com'); 
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The site 10.0.1.2:8888 wants to modify an existing cookie. 
全 OD Use my choice for all cookies from this site 


Bllow [alow for Session | | Deny | 


Name; username 
Content; deleted 
Host: 10,0,1,2 
Path: } 
Send For: Any type of connection 








Expires: Sunday, December 20, 2009 5:04:00 PM 





图 9-13 ” 当 发 送 删除 cookie 时 ，Firefox 显 示 的 cookie 信 息 


诠释 这 个 特性 ， 我 们 为 Web 应 用 程序 添加 一 个 重 置 页 面 ， 它 将 摧毁 已 发 送 的 cookie 以 删除 用 
户 的 偏好 设置 。 


这 删除 一 个 cookie 
(1) 在 文本 编辑 器 或 者 IDE 中 开启 一 个 新 的 PHP 脚 本 ， 命 名 为 reset .php (参看 脚本 9-4): 


<?php // 脚本 9-4 - reset.php 








脚本 9-4 ”发 送 同 已 有 cookie 同 名 的 空白 cookie 和 过 去 的 过 期 时 间 ， 重 设 所 有 的 cookie 的 值 
<?php // 脚本 9-4 - reset.php 
// 删除 cookie: 


1 

2 

3 

4 setcookie('font size', '', time()- 600, '/'); 
5 setcookie('font color', '', time()- 600, '/'); 
6 

7 

8 

9 





?><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 











10 <head> 

下 由 <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 9 

;2 <title>Reset Your Settings</title> 

13 </head> 

14 <body> 

3 

16 <p>Your settings have been reset! Click <a href="view settings.php">here</a> to 
go back to the main page.</p> 

17 

18 </body> 

19 </html> 

(2) 使 用 发 送 空白 cookie 的 方式 删除 已 有 的 cookie， 并 完成 PHP 代 码 : 

setcookie('font size', '', time() - 600, '/'); 

setcookie('font color', '', time() - 600, '/'); 

?> 


行 代码 发 送 了 名 为 font_size 和 font_color 的 两 个 cookie， 两 个 都 没有 值 ， 并 且 过 期 
时 间 都 在 10 分 钟 以 前 。 正 如 在 创建 cookie 时 那样 ， 必 须 在 任何 其 他 信息 被 发 送 至 Web 浏 览 器 之 前 
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调用 函数 setcookie()。 
(3) 创建 HTML 的 head 部 分 : 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Reset Your Settings</title> 

</head> 


(4) 添加 页 面 的 body 部 分 : 


<body> 
<p>Your settings have been reset! Click <a href="view settings.php"> 
一 here</a> to go back to the main page.</p> 

















</body> 

脚本 中 的 body 部 分 仅仅 用 来 告诉 用 户 他 们 的 设置 已 经 被 重 置 。 现 在 提供 了 一 个 链接 来 引导 
他 们 返回 到 主页 。 

(5) 完成 HIML: 

</htmil> 


(6) 将 页 面 保存 为 reset .php， 放 置 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 ， 并 在 浏览 器 中 
测试 (参见 图 9-14、 图 9-15 和 图 9-16)。 


The site 10.0.1.2:8888 wants to modify an existing cookie. 





[DD] Use my choice For all cookies From this site 


Hide Details 是 low For Session 


Name: Font_size 
Content; deleted 
Host; 10,0,1,2 
Path; |/ 
Send For: Any type of connection 
Expires: Sunday, December 20, 2009 5:;08:01 PM 











图 9-14 ” 当 对 只 有 名 称 没 有 值 的 cookie 使 用 函数 setcookie() 时 ,已 经 存在 的 同名 
cookie 将 被 删除 。 设 置 为 过 去 的 过 期 时 间 也 将 会 确保 正确 删除 已 存在 的 cookie 





岛 国电 


Your settings have been reset! Click here to go 
back to the main page. 








图 9-15 ” 重 置 页 面 发 送 两 个 空 cookie 并 将 显示 这 样 的 消息 
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琅 四 Ey 


Customize Your Settings 


Reset Your Settings 


yadda yadda yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda yadda yadda 
yadda yadda yadda yadda yadda yadda yadda 
yadda yadda yadda yadda 











| 








9-16 如 果 用 户 使 用 了 这 个 重 置 页 面 ，PHP 将 摧毁 这 个 cookie 
(参见 图 9-14) ， 并 将 重 置 主页 为 它 默 认 的 格式 


单 击 view_settings .php (参看 脚本 9-2) 中 适当 的 链接 ,测试 该 页 面 ,或 者 直接 前 往 该 














Vv 提示 

口 在 加 载 另 外 的 页 面 之 后 cookie 才 会 生效 。 类 似 地 ， 删 除 一 个 cookie 也 需要 等 到 加 载 另 外 的 
页 面 之 后 。 这 就 是 说 可 以 删除 一 个 页 面 中 的 cookie, 但 是 仍然 可 以 对 之 进行 访问 (这 是 因 
为 页 面 获 取 cookie 是 在 删除 cookie 之 前 )。 

口 如 同 使 用 不 同 的 浏览 器 将 会 呈现 创建 cookie 的 不 同 结果 那样 ， 删 除 cookie 也 会 造成 同样 的 
状况 。 请 在 多 个 浏览 器 中 测试 脚本 ， 并 且 使 用 setcookie () 设置 以 确保 最 佳 最 全 面 的 兼 








9.6 什么 是 session 


Session 就 像 是 cookie， 它 是 一 种 解决 方案 ， 可 以 跟踪 用 户 在 一 系列 页 面 中 访问 的 数据 。 两 者 
之 间 非 常 重要 的 区 别 在 于 ，cookie 将 数据 保存 在 客户 端 (在 Web 浏 览 器 中 )， 而 session 则 将 数据 保 
存在 服务 器 端 。 因 为 有 这 个 区 别 ，session 拥 有 比 cookie 更 多 的 优势 。 
口 session 通 常情 et 全 ， 这 是 因为 数据 不 会 在 客户 端 和 服务 器 端 来 回 重复 传输 。 
口 Session 能 够 存储 比 cookie 更 多 的 信息 
口 session 甚 至 可 以 在 用 户 不 接受 他 们 济 览 嚣 中 的 cookie 时 仍然 能 够 工作 。 

当 开 启 一 个 session 时 ，PHP 将 会 创建 一 个 随机 的 session ID。 每 个 用 户 的 session 都 会 有 一 个 自 
己 的 session ID ， 与 服务 器 上 存储 该 用 户 session 数 据 的 文本 文件 名 相同 (参看 脚本 9-5 ) 。 

这 样 网 站 上 所 有 PHP 脚 本 都 可 能 使 用 某 一 用 户 的 session 数 据 ， 同 时 还 会 追踪 session ID 。 默 认 
情况 下 ， 这 个 session ID 将 作为 一 个 cookie 发 送 给 Web 浏 览 器 (参见 图 9-17)。 接 下 来 PHP 页 面 将 使 
用 这 个 cookie 来 检索 session ID 并 访问 session 信 息 。 
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四 


The site 10.0.1.2:8888 wants to set a cookie. 
全 回 ] Use my choice For all cookies from this site 


Hide Details Allow For Session Deny 


Name: 
Content: 
Host' 
Path' 
Send For; 
Expires: 











PHPSESSID 
2al4449d659F29F9bd6F00a6886c78Fb 
.Di 之 

1 

Any type of connection 

At end of session 


在 下 面 儿 页 中 ， 将 看 到 使 用 PHP 可 以 非常 容易 地 操作 session 。 


在 session 和 cookie 中 进行 选择 


图 9-17 一 个 带 有 session 信 息 的 cookie 被 发 送 给 Web 浏 览 器 


session[] cookiel OUIU00UUUUUUUU cied 00 0 


DDD session UO0DO 
oD00000U0000 





o00000000000000 
o000000000000000 


D0000000000000000000000000U00UUUUUULU cookieb 
DO00000000000000000000U0U0U00U sessiond 0 UUD sessiong 0 0 


国有 


9.7 创建 session 


可 以 使 用 函数 session_start () 创 建 、 访 问 或 删除 session。 这 个 函数 将 试图 在 session 首 次 





启动 时 发 送 一 个 cookie， 因 此 它 必须 在 任何 HTML 或 空白 被 发 送 至 Web 浏 


I 口 S 一 


册 秦 之 


在 使 用 session 的 页 面 中 ， 必 须 在 脚本 的 起 始 行 调 用 图 数 session_start() : 


<?php 
session_start(); 


前 调用 。 





因此 ， 


当 第 一 次 开启 session 时 ， 会 产生 一 个 随机 的 session ID ， 并 且 会 向 Web 浏 览 器 发 送 一 个 名 为 
PHPSESSID (session 的 名 称 ) 的 cookie, 它 的 值 看 上 去 类 似 于 4bce48dc87cb4b54d63f99da23fb41lel 


(参见 图 9-18)。 


一 旦 启用 session， 可 以 通过 向 数组 $_sESsION 赋 值 的 方式 记录 数据 : 


$_SESSION['first name'] = 'Sam'; 





$_SESSION['age'] = 4; 


这 个 数组 与 PHP 中 使 用 的 其 他 数组 不 同 ， 它 必须 是 关联 数组 。 换 句 话 说 ， 应 该 显 式 地 使 用 字 
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符 串 作为 键 ， 如 fizrst_name 和 age。 
每 次 这 样 做 时 ，PHP 将 向 服务 器 上 的 一 个 临时 文件 中 编写 一 些 数 据 (参看 脚本 9-5 ) 。 


脚本 9-5 session 数据 在 服务 器 文件 中 的 存储 方式 
1 email|s:14:"me@example.com"; loggedin|i:1292883103; 


首先 我 们 将 重 写 第 8 章 中 的 登录 脚本 ， 这 次 要 将 Email 地 址 保存 在 一 个 session 中 。 


(1) 在 文本 编辑 器 或 者 IDE 中 打开 login.php (参看 脚本 8-13) 。 
(C) 在 cb_endq_clean () 代码 行 之 前 ， 添 加 下 面 的 代码 (参看 脚本 9-6): 


session_start(); 


$_SESSION['email'] = $_POST['email']; 
$_SESSION['loggedin'] = time(); 


脚本 9-6 该 脚本 保存 了 session 中 的 两 个 值 ， 并 且 将 用 户 重新 定向 至 其 session 值 被 访问 的 其 他 页 面 









































1 <?php // 脚本 9-6 - login.php #3 

2  /* 页 面 可 以 用 于 用 户 登 录 (基本 完成 |)。*/ 

3 

4 ” // 设置 页 面 标题 并 包含 页 头 文 件 : 

5 define('TITLE', 'Login'); 

6 include('templates/header.html'); 

7 

8 ”// 打印 一 些 介绍 文本 : 

9 print '<h2>Login Form</h2> 

10 <p>Users who are logged in can take advantage of certain features like this, that, 
and the other thing.</p>'; 

二 再 

12 // 检查 表单 是 否 已 提交 : 

13 if ($_SERVER['REQUEST METHOD'] == 'POST') { 

14 

15 // 处 理 表单 : 

16 if ( (iempty($_POST['email'])) && (!Iempty(S_POST['password'])) ) { 

17 

18 if ( (strtolower($_ POST ['email']) == 'me@example.com') && ($_POST['password'] == 

'testpass') ) { // 相等 ! 

19 

20 // 开局 session 缓冲 : 

21 session start(); 

22 $_SESSION['email'] = $_ POST['email']; 

23 $_SESSION['loggedin'] = time(); 

24 

25 // 将 用 户 重 定向 到 欢迎 页 ! 

26 ob_end_clean(); // 消除 缓冲 ! 

27 header ('Location: welcome.php'); 

28 二 区 二 EC) 

29 

30 } else { // 不 相等 ! 

31 

32 print '<p>The submitted email address and password do not match those on 

file!<br/>Go back and try again.</p>'; 
33 


230 0 90 cookie[l] session 








35 

36 } else { // 表单 未 填写 完整 。 

37 

38 print '<p>Please make sure you enter both an email address and a password!<br 
/>Go back and try again.</p>'; 

39 

40 } 

41 

42 } else { // 显示 表单 。 

43 

44 print '<form action="login.php" method="post"> 

45 <p>Email Address: <input type="text" name="email" size="20" /></p> 

46 <p>Password: <input type="password" name="password" size="20" /></p> 

47 <p><input type="submit" name= "submit" value="Log In!" /></p> 

48 </form>'; 

49 

5.0:, `} 

与 小 

52 include('templates/footer.html'); // 包含 页 脚 文 件 。 

537 2%> 


可 以 通过 调用 函数 session_start () 以 在 session 中 保存 值 。 虽 然 通常 必须 在 脚本 中 首先 调 
用 此 函数 (因为 它 将 试图 发 送 一 个 cookie), 但 是 在 这 里 并 不 是 必须 的 , 这 是 因为 该 脚本 的 header 
文件 开启 了 输出 缓存 (参见 第 8 章 )。 

session 将 首先 在 $_SESSION['email'] 中 保存 用 户 提交 的 Email 地 址 。 然 后 将 用 户 登 录 的 时 
间 赋 值 给 $_sESSION['loggedin']。 这 是 由 调用 函数 time () 所 决定 的 , 它 将 返回 从 epoch (1970 
年 1 月 1 日 午夜 ) 开始 过 去 的 秒 数 。 

(3) 将 文件 保存 为 login .php， 放 置 在 启用 了 PHP 的 计算 机 上 适当 的 目录 中 。 

该 脚本 应 该 被 放置 在 同 第 8 章 中 所 述 相同 的 目录 中 ， 因 为 它 需 要 其 中 一 些 其 他 的 文件 。 

(4) 在 Web 浏 览 器 中 加 载 该 表单 以 确保 它 没有 错误 发 生 (参见 图 9-18)。 


加 回避 加 | 

















Raise High the Roof Beam! 


Favorite 
Login Form 
1 don't exactly k 


mean it 
in can take advantage of certain features like this, = The Catchery 





1privately say t 
accept from me 
early-blooming 
- Raise High the 
Sepmour An nr 


5:11 pm Monday [ 














图 9-18 登录 表单 
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没有 完成 和 提交 登录 表单 ， 因 为 欢迎 页 面 需要 在 实际 登录 之 前 更 新 。 

V 提示 

口 php.ini 配 置 文件 包含 许多 同 session 相 关 的 设置 , 如 果 具 有 服务 器 管理 员 级 别 的 权限 , 就 

可 以 修改 php .ini 文 件 。 在 文本 编辑 器 中 打开 php .ini 文 件 以 查看 提供 更 多 信息 的 手册 。 

口 还 可 以 使 用 ini_set () 函数 来 更 改 一 些 session 设 置 。 

口 session_name () 函数 可 以 修改 session 的 名 称 ( 替 代 上 默认 的 PHPSESSID)。 它 必须 在 每 次 
调用 session_start () 之 前 使 用 ， 就 像 这 样 : 


session name('YourVisit'); 
session_start(); 


口 session_ set _cookie _params () 畏 数 用 来 修改 session cookie 的 设置 (如 过 期 时 间 、 路 
径 和 域 ) 。 

口 常量 SID 用 格式 name=ID 的 方式 保存 一 个 字符 串 。 例 如 : 
PHPSESSID=4bcc48dc87cb4b54d63f99da23fb41lel 

口 可 以 在 session 中 保存 任何 类 型 的 值 ， 如 数值 、 字 符 串 、 数 组 或 者 对 象 ， 甚 至 是 它们 的 任意 
组 合 。 





























9.8 访问 session 变量 


我 们 已 经 在 session 中 保存 了 值 ， 接 下 来 需要 学 习 如 何 访问 这 些 值 。 不 论 是 创建 了 新 的 session 
或 者 访问 一 个 已 经 存在 的 session ， 都 必须 从 函数 session_start () 开始 。 该 函数 向 PHP 指 明 这 
个 特殊 的 脚本 将 要 使 用 session。 

引用 $_sesston 变 量 非常 简单 ， 和 引用 其 他 数组 的 操作 差不多 。 考 虑 到 这 一 点 ,这 里 将 编写 
另外 一 个 同 第 8 章 类 似 的 欢迎 页 面 ， 用 来 访问 存储 的 Email 和 1oggedin 的 值 。 











信访 问 session 变 量 的 步骤 











(1) 在 文本 编辑 器 或 者 IDE 中 创建 一 个 新 的 PHP 文 档 ， 命 名 为 welcome .php (参看 脚本 9-7): 


<?php // 脚本 9-7 - welcome .php 


脚本 9-7 只 要 脚本 首先 使 用 session_start ()， 就 可 以 通过 使 用 数组 $_sESSION 来 访问 已 保 
存 的 session 值 

<?php // 脚本 9-7 - welcome.php #2 

/* 这 是 欢迎 页 ， 用 户 成 功 登 录 后 

会 被 重 定向 到 这 里 。*/ 


// 开始 session 部 分 : 
session start(); 


// 设置 页 面 标题 并 包含 头 文件 : 
define('TITLE', 'Welcome to the J.D. Salinger Fan Club!'); 


\D oemn 心 wmN 
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10 include('templates/header.html'); 
下 二 
12 // 打印 欢迎 信息 : 
13 print '<h2>Welcome to the J.D. Salinger Fan Club!</h2>'; 
14 print '<p>Hello, ' . $ SESSION ['email'] . '!</p>'; 
15 
16 // 打印 用 户 登 录 时 间 : 
17 date_ default timezone_set('America/New_York'); 
18 print '<p>You have been logged in since: ' . date('g:i a', $_ SESSION['loggedin']). 
'</p>'; 
19 
20 // 创建 登 出 链接 : 
21 print '<p><a href="logout.php">Click here to logout.</a></p>'; 
22 
23 include('templates/footer.html'); // 包含 页 脚 文件 。 
24 ?> 


(2) 开始 编写 session 部 分 : 
session_ start(); 


当 访 问 session 值 的 时 候 ， 应 该 在 将 任何 数据 发 送 给 Web 训 览 器 之 前 调用 函数 session_ 





start(), 
(3) 定义 页 面 的 标题 ， 并 且 包 含 HTML 的 header 部 分 : 
define('TITLE', 'Welcome to the J.D. Salinger Fan Club!'); 





include('templates/header.html'); 


由 于 该 页 使 用 的 模板 系统 同 第 8 章 中 所 开发 的 相同 ， 因 此 它 也 将 使 用 同样 的 header 系 统 。 
(4) 使 用 用 户 的 Email 地 址 显示 问候 信息 : 


print '<h2>Welcome to the J.D. Salinger Fan Club!</h2>'; 
print '<p>Hello, ' . $_SESSION ['email'] . '!</p>'; 


可 以 通过 引用 $_SESSION['email'] 来 访问 已 保存 的 用 户 地 址 。 在 这 里 ， 该 值 已 同 正在 打 
印 的 字符 串 剩余 部 分 连接 。 























(5) 显示 用 户 已 经 登录 的 时 长 : 


date_default_timezone_set ('America/New_York'); 
print '<p>You have been logged in since: ' . date('g:i a', $_SESSION['loggedin']) . '</p>'; 


可 以 通过 引用 $_SESSION['loggedin'] 变 量 的 方式 显示 用 户 已 经 登录 的 时 长 。 通 过 将 之 作 











给 date() 函数 的 第 二 个 参数 ， 连 同 具 有 适当 格式 的 参数 ， 可 以 使 用 PHP 脚 本 创建 类 似 于 


11:22pm 这 样 的 文本 。 
但 是 在 使 用 aate () 函数 之 前 ， 需 要 设 定 默认 时 区 (这 点 在 第 8 章 中 也 讨论 过 )。 如 果 愿 意 ， 








在 设置 了 时 区 之 后 ， 可 以 从 页 脚 文件 中 移 除 对 该 函数 的 调用 。 


(6) 完成 内 容 部 分 


print '<p><a href="logout.php">Click here to logout.</a></p>'; 


下 一 


段 脚 本 将 提供 注销 功能 ， 因 此 这 里 需要 添加 一 个 链接 。 


0) 将 HTML 页 脚 包含 进来 ， 并 且 完 成 HTML 页 面 : 
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require('templates/footer.html'); 
pe 


(8) 将 文件 保存 为 velcome .php， 放 置 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 ， 并 且 在 Web 浏 
览 器 中 (参见 图 9-19) 测试 (从 脚本 9-6 中 1ogin.php 开 始 )。 








elspme to the J.D. Salinger Fan 
lub! 


Hello, me@example.com! 


You have been logged in since: 5:11 pm 





Click here to logout. 





图 9-19 ”在 成 功 进行 登录 之 后 (在 表单 中 使 用 me@example.com 和 ltestpass)， 
用 户 被 重新 定向 至 本 页 ， 在 这 里 将 使 用 session 的 值 对 他 们 进行 问候 

















V 提示 
口 可 以 使 用 isset ($_SESSION['var']) 来 查看 特定 的 session 值 是 否 存 在 ， 就 像 用 来 检验 
其 他 的 变量 是 否 被 设置 那样 。 
口 在 session 中 ， 数 据 都 会 以 纯 文本 的 形式 保存 在 一 个 开放 可 读 的 文本 文件 中 。 请 时 刻 注意 
session 中 保存 的 数据 ， 永 远 不 要 将 真实 的 敏感 信息 存在 session 中 ， 如 信用 卡 数据 。 
口 要 提高 安全 性 ， 可 以 将 数据 加 密 后 再 保存 到 session 中 ,在 读 取 session 后 再 将 其 解密 。 

要 使 用 Mcrypt 库 和 一 些 高 级 的 PHP 知 识 。 


9.9 删除 session 9 


知道 如 何 删除 session 是 非常 重要 的 ， 就 像 知道 如 何 删 除 cookie 一 样 重 要 ， 有 时 候 需 要 删除 已 
经 保存 的 session 数 据 。session 的 数据 在 两 个 地 方 存在 ， 因 此 需要 在 两 个 地 方 进行 删除 操作 。 但 是 
首先 必须 从 函数 session_start () 开始 ， 通 常 像 下 面 这 样 : 

Session start(); 

然后 ， 再 次 设置 数组 $_sESSION 来 删除 session 的 值 ; 

$_SESSION = array(); 

最 后 ， 需 要 从 服务 器 上 删除 session 数 据 (保存 在 临时 文件 中 )。 可 以 通过 下 面 的 方式 进行 操 
作 : 

session destroy(); 


我 们 编写 logout .php， 它 将 用 来 删除 session， 以 有 效 地 注销 用 户 。 
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删除 session 
(1) 在 文本 编辑 器 或 者 IDE 中 新 建 一 个 PHP 脚 本 ， 命 名 为 1ogout .php (参看 脚本 9-8)。 


<?php // 脚本 9-8 - logout.php 











脚本 9-8 删除 session 需 要 这 样 的 3 个 步骤 : 开启 session， 重 新 设置 数组 ， 然 后 删除 session 数 据 


<?php // 脚本 9-8 - logout.php 
/* 登 出 页 面 ， 会 删除 session 数据 。*/ 


// 开始 Session 部 分 : 
session start(); 


// 删除 Session 变量 : 
unset ($_SESSION) 7 


\Do~aw 必 wmN 情 


// 重 置 Session 数 组 : 
$_SESSION = array(); 


js 才 


o auwm 必 wbP ce 


// 定义 页 面 标题 并 包含 页 头 文件 : 
define('TITLE', 'Logout'); 
include('templates/header.html'); 


汪汪 





Ko) 


<h2>Welcome to the J.D. Salinger Fan Club!</h2> 

<p>You are now logged out.</p> 

<p>Thank you for using this site. We hope that you liked it.<br/> 
Blah, blahy blalh,.d,. 

Blah, blah, blah...</p> 


NDNDDNDINNAI 
心 wN 上 吕 


25 <?php include('templates/footer.html'); ?> 

(2) 开始 编写 session 部 分 : 

session_start(); 

请 记 住 ， 直 到 用 这 个 函数 激活 session 后 才能 对 session 进 行 删除 。 

(3) 重 置 session 数 组 

$_SESSION = array(); 

在 第 7 章 介 绍 过 ，array () 函数 会 创建 一 个 新 的 空 数 组 。 将 这 个 函数 调用 的 结果 赋 给 
$_SESSION，$_SESSION 中 所 有 的 0 -0 对 都 会 被 删除 。 

(4) 删除 服务 器 上 的 session 数 据 : 

session_ destroy(); 

该 步骤 告知 PHP 将 服务 器 上 真正 的 session 文 件 删除 。 

(5) 将 HTML 的 header 部 分 包含 进来 ， 并 且 完 成 PHP 部 分 : 


define('TITLE', 'Logout'); 
include('templates/header.html'); 
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?> 
2 [pb 公立 
(6) 完成 页 面 的 内 容 部 分 : 
<h2>Welcome to the J.D. Salinger Fan Club!</h2> 
<p>You are now logged out.</p> 
<p>Thank you for using this site. We hope that you liked it.<br/> 


Blah, blah, blah... 
Blah, blah, blah...</p> 


(7) 将 HTML 页 脚 包 含 进来 : 
<?php require ('templates/footer.html'); ?> 


(8) 将 文件 保存 为 logout .php， 放置 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 ,并 且 通 过 单 击 
welcome .php 上 的 链接 来 在 Web 浏 览 器 上 测试 (参见 图 9-20)。 





WelSpme to the J.D. Salinger Fan 
lub! 


You are now logged out 


Thank You for using this site. We hope that YOU liked it 





Blah, blah, blah... Blah, blah, blah 














图 9-20 ”注销 页 面 删除 了 session 数 据 








ww 提示 

口 可 以 使 用 unset ($s_SESSION['var']) 来 删除 一 个 单独 的 session 值 。 

口 服务 器 上 的 PHP 模 块 将 自动 依照 它 配置 中 的 设置 执行 垃圾 回收 。 PHP 假 定 这 些 session 文 件 9 
不 再 需要 ， 使 用 垃圾 回收 以 手动 从 服务 器 上 删除 它们 。 

口 当 session ID 必须 附加 给 站 点 中 每 个 链接 (以 便 让 每 个 页 面 都 能 够 接受 到 session ID ) 的 情 
况 下 ,可 以 让 PHP 使 用 session 而 不 使 用 cookie。 如 果 启 用 了 enable_trans_side 设 置 (在 
php .ini 文 件 中 )，PHP 将 为 对 此 进行 处 理 。 


9.10 回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
comy/forumy/ 。 























9.10.1 回顾 
口 cookie 在 什么 地 方 存 储 数据 ? session 在 什么 地 方 存 储 数据 ?哪个 更 安全 ? 
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口 说 出 两 个 处 理 cookie 问 题 的 调试 技术 。 

D setcookie () 函数 的 两 个 参数 path 和 domain 是 如 何 限制 cookie 访 问 方式 的 ? 
口 如 何 删 除 一 个 cookie? 

口 如 果 要 创建 或 访问 session 数 据 ， 页 面 必须 调用 哪个 函数 ? 

口 为 什么 session 会 使 用 cookie (默认 情况 下 ) ? 


























9.10.2 ”实践 


口 如 果 还 未 安装 Firebug 扩 展 包 ， 在 Firefox 中 安装 它 。 前 提 是 先 要 安装 Firefox。 

口 在 PHP 手 册 中 查找 setcookie () 函数 。 了 解 关 于 操作 cookie 的 其 他 说 明 信 息 和 用 户 注释 。 

口 重 写 customize.php， 以 便 让 该 脚本 也 能 应 用 用 户 人 和 偏好。 提示: 应 考虑 到 cookie 在 设置 
后 不 会 立即 生效 。 在 表单 提交 之 后 ， 应 该 使 用 $_GET 值 编写 CSS 代 码 ， 第 一 次 打开 页 面 使 
用 $_COOKIE (如 果 cookie 已 经 存在 ) ， 否 则 使 用 默认 值 。 

口 将 customize.php 页 面 中 的 表单 改 为 粘性 表单 ， 使 其 显示 用 户 的 当前 选择 。 

口 重 写 welcome .php， 在 print 语 名 中 使 用 双 引 号 打印 出 用 户 Email 地 址 和 欢迎 信息 。 

口 这 次 加 大 难度 ， 重 写 welcome .php， 在 print 语 句 中 使 用 双 引 号 打印 出 用 户 的 登录 时 长 。 

提示 : 需要 使 用 变量 。 

口 重 写 最 后 三 个 脚本 ， 让 session 使 用 自 定 义 名 称 。 
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本 章 内 容 

口 创建 和 使 用 简单 函数 

口 创建 和 调用 接受 参数 的 函数 
口 设置 默认 参数 值 

口 创建 和 使 用 带 有 返回 值 的 函数 
口 理解 变量 作用 域 

口 回顾 和 实践 








在 本 书 中 , 我 们 已 经 使 用 了 几 十 个 函数 (如 date()、setcookie() 和 number_format () )， 
它们 都 提供 了 非常 有 用 的 功能 。 尽 管 这 些 函 数 都 已 经 被 PHP 定 义 过 , 但 是 在 这 里 我 们 将 创建 自己 
的 函数 。 需 要 注意 的 是 ， 函 数 创 建 之 后 ， 可 以 像 使 用 PHP 内 置 的 函数 那样 使 用 它们 。 

创建 函数 能 够 为 编程 节省 大 量 的 时 间 。 事 实 上 ， 它 们 是 创建 Web 应 用 程序 和 生成 在 将 来 的 项 
目 中 使 用 的 可 靠 的 PHP 代 码 库 过 程 中 重要 的 一 步 。 

本 章 将 介绍 如 何 编写 自己 的 函数 以 执行 特定 的 任务 。 随 后 ， 将 讲授 如 何 向 函数 传递 信息 ， 
在 函数 中 使 用 默认 的 值 ， 以 及 让 函数 返回 值 。 此 外 ， 还 将 介绍 函数 和 变量 是 如 何 共同 工作 的 。 


10.1 创建 和 使 用 简单 函数 


在 编程 时 , 你 将 会 发 现 无 论 是 在 一 段 单独 的 脚本 内 还 是 在 儿 段 脚本 的 执行 过 程 中 , 都 会 频繁 
地 使 用 某 些 特定 的 代码 段 。 将 这 些 例 程 作为 自 定义 的 函数 将 节省 大 量 时 间 , 并 且 可 以 让 编程 过 程 
更 加 人 简便， 尤其 是 当 Web 站 点 变 得 越 来 越 复杂 时 。 如 果 创 建 一 个 函数 ， 该 函数 就 可 以 在 每 次 被 调 
用 时 发 生 相 应 的 行为 ， 类 似 于 Print 在 每 次 使 用 时 都 会 向 浏览 器 发 送 文本 。 

创建 一 个 用 户 自 定 义 函数 的 语法 如 下 : 


function function name () { 
statement(s); 














} 
例如 : 
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function whatever() { 
print 'whatever'; 


} 

可 以 采取 与 为 变量 命名 大 致 相同 的 规则 来 为 函数 命名 ,只 是 不 需要 以 美元 符号 作为 开头 。 其 
次 ， 建 议 创建 的 函数 名 称 是 有 意义 的 ， 就 像 应 当 编 写 有 代表 性 意义 的 变量 名 称 那 样 (create_ 
header 作 为 函数 名 将 比 Eunction1 要 好 得 多 )。 请 记 住 , 不 要 使 用 空格 ( 即 不 能 使 用 两 个 分 隔 的 
词 作为 函数 名 称 )， 这 将 会 导致 产生 错误 消息 (可 以 使 用 下 划 线 来 替代 空格 )。 不 同 于 变量 的 是 ， 
PHP 中 的 函数 名 不 区 分 大 小 写 ， 但 是 仍然 需要 坚持 命名 方法 的 一 致 性 。 

任何 有 效 的 PHP 代 码 都 能 够 在 函数 的 statement (s) 区域 运 行 ， 包括 对 其 他 函数 的 调用 。 对 
于 函数 所 包含 的 语句 数量 并 没有 限制 , 但 是 请 确定 每 个 语句 的 结尾 都 使 用 分 号 ,就 像 在 PHP 脚 本 
的 其 他 部 分 那样 。 函 数 也 可 以 包含 任意 控制 结构 的 组 合 ， 条 件 或 循环 。 

只 要 必需 的 要 素 都 齐全 ， 函 数 的 格式 并 不 是 那么 重要 。 这 些 要 素 包 括 function 这 个 词 、 函 
数 名 称 、 打 开 和 关闭 的 圆 括号 、 打 开 和 关闭 的 花 括号 以 及 语句 。 按 照 惯 例 ， 函 数 的 语句 通常 都 会 
相对 于 function 关 键 字 那 行进 行 缩 进 ， 以 便 看 起 来 更 加 明晰 ， 就 像 在 循环 和 条 件 中 所 作 的 那样 。 
在 任何 情况 下 ， 你 都 要 选择 一 种 喜欢 的 格式 样式 〈 语 法 正确 且 合乎 逻辑 ) 并 且 坚持 一 直 使 用 它 。 

可 以 通过 引用 的 方式 对 函数 进行 调用 ， 就 像 调 用 内 置 函数 那样 。 下 面 这 行 代码 : 

whatever (); 

将 会 执行 前 面 定义 的 函数 语句 部 分 中 的 print 语 句 。 
首先 创建 一 个 函数 ， 用 来 生成 表单 中 的 月 、 日 和 年 的 下 拉 菜 单 (参见 图 10-1)。 


anuay _ 1) 1 WEEE 







































































图 10-1 这些 下 拉 菜 单 是 使 用 一 个 用 户 自 定义 函数 所 创建 的 。 这 项 技术 
将 会 让 代码 更 具 移 植 性 而 不 影响 最 终 的 结果 








> 创建 和 调用 基本 函数 
(1) 在 文本 编辑 器 或 者 IDE 中 创建 一 个 新 的 PHP 文 档 ， 命 名 为 menus .php (参看 脚本 10-1): 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

















"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
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<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Date Menus</title> 


</head> 
<body> 


脚本 10-1 脚本 中 定义 的 函数 为 表单 创建 了 3 个 下 拉 菜 单 


1 
2 
3 
4 
5 
6 
a 
8 
9 


ON 人 VOPO 





40 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Date Menus</title> 

</head> 

<body> 

<?php // 脚本 10-1 - menus.php 

/* 脚本 定义 并 调用 函数 。*/ 








// 函数 创建 3 个 下 拉 菜 单 :月 、 上 日、 年 。 


function make date menus() { 


// 将 月 以 数组 形式 存储 : 
Smonths = array (1 => 'January', 'February', 'March', 'April', 'May', 'June' 
'July', 'August', 'September', 'October', 'November', 'December'); 


// 创建 月 下 拉 菜 单 : 
print '<select name="month">'; 
foreach (Smonths as Skey => $value) { 
print "\n<option value=\"$key\">$value</option>"; 
} 


print '</select>'; 


// 创建 日 下 拉 菜 单 : 
print '<select name="day">'; 
for ($day = 1; $day <= 31; $day++) { 
print "\n<option value=\"$day\">$day</option>"; 
} 


print '</select>'; 


// 创建 年 下 拉 菜 单 : 

print '<select name="year">'; 

SSstart_year = date('Y'); 

for ($y = $start year; S$y <= ($start _ year + 10); Sy++) { 
print "\n<option value=\"$y\">$y</option>"; 

: 


print '</select>'; 
} // 结束 make_date_menus () 阵 数 。 


// 创建 表单 : 

print '<form action="" method="post">'; 
make_date menus () ; 

rint, TZ/EoOrm> Ys; 
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47 ?> 
48 </body> 
49 </html> 


(2) 开始 PHP 代 码 部 分 : 

<?php // 脚本 10-1 - menus .php 

G) 开始 定义 函数 ， 

function make _ date menus() { 

这 个 函数 的 名 称 为 make_date_menus， 它 在 描述 函数 作用 的 同时 还 非常 易于 记忆 。 
(4) 创建 month 下 拉 菜 单 : 


Smonths = array (1 => 'January', 'February', 'March', 'April', 'May', 'June', 'July', 
»'August', 'September', 'October','November', 'December'); 

print '<select name="month">'; 

foreach (Smonths as S$Skey => $value) { 
print "\n<option value=\"$key\">$value</option>"; 

} 


print '</select>'; 


生成 一 个 月 份 的 列表 ， 首 先 要 创建 一 个 月 份 名 称 的 数组 ， 数 值 类 型 的 索引 从 1 开始 ， 代 表 
January。 当 为 第 一 个 数组 元 素 指明 索引 时 ， 其 余 的 元 素 将 会 跟随 延续 索引 编号 ， 而 不 用 分 别 指 
明 。 





= 





在 数组 创建 之 后 ， 将 打印 出 初始 select 标 签 。 然 后 ， 在 数组 Smonths 上 运行 foreach 循 环 。 
对 于 数组 中 的 每 一 个 元 素 , 打 印 HIML option 标 签 ,并 且 以 数组 的 键 (从 数字 1 到 12) 作 为 option 
的 值 ， 同 时 以 数组 的 值 (January 到 December) 作为 显示 文本 。 每 行 都 用 一 个 换行 符 〈\n) 开 
头 以 便 在 HIML 源 代码 中 每 个 option 都 单独 成 行 。 

(5) 创建 aay 下 拉 菜 单 : 

print '<select name="day">'; 

for ($day = 1; S$day <= 31; S$day++) { 

print "\n<option value=\"$day\">$day</option>"; 


} 


print '</select>'; 

创建 day 菜 单 是 非常 容易 的 。 可 以 使 用 一 个 简单 的 从 数字 1~31 的 for 循 环 来 做 到 。 

(6) 创建 year 下 拉 菜 单 : 

print '<select name="year">'; 

$sstart _ year = date ('Y'); 

for ($y = $start year; Sy <= ($start year + 10); Sy++) { 

print "\n<option value=\"$y\">$y </option>"; 

} 

print '</select>'; 

为 了 创建 year 下 拉 菜 单 ， 将 使 用 date () 国 数 来 获取 当前 的 年 份 。 然 后 用 for 循 环 为 这 个 年 
份 创 建 接 下 来 下 10 个 年 份 的 选项 。 

(7) 结束 函数 : 
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} // 结束 make_date_menus () 巴 数 。 


当 创建 函数 时 , 很 容易 会 因为 遗忘 了 关闭 的 花 括号 而 产生 解析 错误 。 你 可 以 通过 添加 注释 来 

















帮助 记 住 这 最 后 一 步 。 


(8) 编写 表单 标签 ， 并 且 调 用 函数 : 


print '<form action="" method="post">'; 
make_date_menus (); 
print: /torn>"; 


Print 语句 用 来 创建 HIML 表 单 标签 。 如 果 不 这 么 做 ，qate 下 拉 莱 单 将 不 能 在 脚本 中 正确 


地 显示 出 来 。 


函数 创建 完成 之 后 ， 就 可 以 通过 函数 名 (不 要 把 名 字 拼 写 错 ) 调用 它 ,注意 在 后 面 加 上 圆 括 


(9) 完成 PHP 和 HTML 代码: 


?> 
</body> 
</html> 


(10) 将 文件 保存 为 nenus .php， 放 置 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 ， 并 且 在 Web 浏 览 


器 上 运行 (参见 图 10-1)。 





(11) 你 可 以 查看 该 页 面 的 HTML 源 代码 ， 看 看 哪些 东西 是 动态 生成 的 (参见 图 10-2)。 





<form action="" method="post"><select name="month"> 
<option value="1">January</option> 

<option value="2">February</option> 

<option value="3">March</option> 

<option value="4">April</option> 

<option value="5">May</option> 

<option value="6">June</option> 

<option value="7">July</option> 

<option value="8">August</option> 

<option value="9">September</option> 

<option value="10">October</option> 

<option value="1]1">November</option> 

<option value="12">December</option></select><select name="day"> 
<option value="1">1</option> 

<option value="2">2</option> 
<option value="3">3</option> 
<option value="4">4</option> 
<option value="5">5</option> 
<option value="6">6</0option> 
<option value="7">7</option> 
<option value="8">8</option> 
<option value="9">9</option> 











图 10-2 ”该 页 面 源 代码 ， 显 示 出 make_date_menus () 函数 中 的 print 语 句 生 成 的 HTML 








Vv 提示 

口 如 果 出 现 some_function.. .错误 消息 ， 这 意味 着 你 正在 调用 一 个 不 存在 的 函数 (参见 
图 10-3)。 这 可 能 是 由 于 函数 名 称 拼写 错误 ， 也 可 能 是 由 于 PHP 版 本 不 支持 该 函数 ， 请 重 
新 查看 PHP 和 手册 。 如 果 在 调用 用 户 自 定义 函数 时 看 到 这 个 错误 ,可 能 是 函数 没有 定义 或 拼 
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错误 所 导致 的 。 重 新 检查 函数 的 定义 和 调用 部 分 的 拼写 ， 查 看 是 否 有 错误 发 生 。 





Fatal error: Call to undefined function make_date_menu0 in 
/Users/larryullman/Sites/phpvqs4/menus.php on line 44 














图 10-3 ”这 个 错误 的 意思 是 PHP 脚 本 没有 找到 指 定名 称 的 函数 定义 。 在 这 个 例子 中 ， 出 错 的 
原因 是 在 国 数 调 用 时 漏 掉 了 “s”: 正确 的 写法 make_dqate_menus () 与 错误 的 写法 


make_date menu() 











口 函数 function_exists() 将 根据 PHP 中 是 否 存在 某 个 函数 而 返回 TRUE 或 者 FALSE。 这 
适用 于 用 户 自 定义 的 函数 和 PHP 内 置 的 函数 : 


if (function exists('some function')) { ... 


口 尽管 在 PHP 中 不 需要 在 调用 函数 之 前 做 预先 定义 ， 但 是 推荐 的 做 法 是 保持 在 脚本 的 开头 
(或 放 在 引入 文件 中 ) 就 定义 好 函数 的 习惯 ， 最 好 不 要 在 脚本 的 结尾 处 定义 。 
D 一 些 人 喜欢 在 他 们 的 函数 中 使 用 这 样 的 语法 : 


function function_ name() 
statement (s); 


} 


口 用 户 自 定义 的 函数 为 PHP 脚 本 增加 了 额外 的 内 存 需 要 , 因此 你 应 该 恰当 地 使 用 它们 。 如 果 
发 现 函数 只 不 过 调用 了 另外 一 个 PHP 函 数 或 者 只 是 一 行 代码 , 就 可 能 没有 很 好 地 利用 自 定 
义 函 数 。 


10.2 创建 和 调用 接受 参数 的 函数 


尽管 创建 一 个 简单 的 函数 是 有 用 的 , 但 更 好 的 作法 是 编写 一 个 能 够 接受 输入 并 且 能 够 依照 输 
入 进行 某 些 操 作 的 函数 。 这 些 函 数 输入 被 称 为 参数 (argument 或 者 parameter)。 这 个 概念 之 前 已 
经 看 到 过 : sort () 函数 接受 一 个 数组 作为 参数 ， 而 函数 正 是 对 这 个 数组 进行 排序 操作 的 。 

编写 接受 参数 的 函数 的 语法 如 下 所 示 : 


function function name($argl, S$arg2, ...){ 
statement(s); 






































} 
函数 的 参数 以 变量 的 形式 存在 , 而 这 些 变量 被 赋予 调用 函数 时 向 函数 发 送 的 值 。 这 些 变量 使 
用 同 PHP 中 其 他 变量 相同 的 命名 规则 进行 定义 : 


function make_ full name ($first, S$last) { 
print S$first . ' ' . $last; 








} 


调用 接受 输入 (参数) 的 函数 和 调用 不 接受 参数 的 函数 非常 类 似 一 一 只 需 记 住 传 递 所 需 的 值 
即 可 。 这 可 以 通过 传递 变量 实现 ， 如 : 


make_ full name ($fn, $1n); 
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或 者 发 送 字符 串 实 现 ， 如 : 

make_full name('Larry', 'Ullman'); 
或 者 结合 上 述 两 种 方法 ， 如 : 

make_full name('Larry', $ln); 

需要 注意 的 是 参数 是 被 逐 字 传送 的 : 在 函数 定义 中 的 第 一 个 变量 将 被 赋予 调用 行 中 的 第 一 个 
值 ， 第 二 个 函数 变量 被 赋予 第 二 个 调用 值 ， 依 此 类 推 〈 参 见 图 10-4) 。 函 数 并 没有 智能 到 能 够 直 
观 地 理解 我 们 对 于 值 之 间 关 联 的 认识 。 如 果 值 传递 失败 ， 函 数 将 假定 其 值 为 空 ( 值 为 空 并 不 等 于 
数值 0， 它 是 真实 存在 的 值 ， 更 接近 于 词语 “无 ")。 同 样 地 ， 如 果 一 个 函数 有 4 个 参数 而 只 传递 了 
3 个 一 一 第 4 个 将 为 空 ， 这 也 会 产生 错误 (参见 图 10-5)。 

















i 
， 站 
function make full néme($first, $las 
Print $first . '/' / glast; 
/ 4 
} 7 Ry 


pr 
make_full_name ($fn7 $1n7; 


/ / -" i 
make_full name('Larry', 'Ullman”); 


2 i 
make full name('Larry', $11); 





图 10-4 ”函数 调用 中 的 值 是 如 何 赋 给 函数 参数 的 








Warning: Missing argument 2 for make_text_input(), called in /Users 
/larryullman/Sites/phpvgs4/sticky 1 .php on line 38 and defined in /Users 
/larryullman/Sites/phpvqs4/sticky1.php on line 14 














图 10-5 在 调用 (用 户 自 定义 或 者 PHP 内 置 的 ) 函数 时 ， 传 递 的 参数 个 数 不 对 将 会 导致 
出 现 错误 消息 

接 下 来 创建 一 个 更 有 意思 的 示例 来 诠释 接受 参数 的 函数 。 在 第 8 章 介绍 了 创建 粘性 表单 ， 使 0 
其 显示 之 前 提交 过 的 值 。 粘 性 文本 框 的 代码 如 下 


First Name: <input type="text" name="first name" size="20" 




















一 Value="<?php if (isset($_ POST ['first name'])) { print htmlspecialchars($_POST 

>['first _ name']); } ?>" /> 

由 于 页 面 中 有 很 多 表单 都 会 反复 使 用 相同 的 代码 ， 例 如 第 8 章 中 的 register .php 因 此 这 里 
非常 适合 使 用 用 户 自 定义 函数 。 





接 下 来 的 脚本 将 定义 并 调用 一 个 函数 , 该 函数 创建 了 烙 性 文本 框 。 函 数 中 的 一 个 参数 用 于 接 
受 文本 框 的 名 字 ， 男 一 个 参数 用 于 接受 文本 框 标签 (文本 提示 )。 这 个 函数 会 被 脚本 多 次 调用 ， 
以 生成 各 种 文本 框 参见 图 10-6)。 表 单 提交 后 ， 表 单 会 记 住 之 前 输入 的 值 。 
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First Name: 





Last Name: 





Email Address: 


( Register! ) 

















妊 


图 10-6 ”以 上 三 个 表单 元 素 是 由 用 户 自 定义 函数 创建 的 








党 创建 和 调用 接受 参数 的 函数 








(GD 在 文本 编辑 器 或 者 IDE 中 创建 一 个 新 的 PHP 文 档 , 命名 为 sticky1 .php (参看 脚本 10-2): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Sticky Text Inputs</title> 

</head> 

<body> 


脚本 10-2 make_text_input () 国 数 接受 两 个 参数 ， 指 明文 本 框 的 名 字 和 标签 














水 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

E: <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>Sticky Text Inputs</title> 

7 </head> 

8 <body> 

9 <?php // 脚本 10-2 - sticky1.php 





10 /* 脚本 定义 并 调用 函数 ， 该 函数 创建 粘性 文本 框 。*/ 

11 

12 // 函数 创建 粘性 文本 框 。 

13 // 函数 接受 两 个 参数 。 

14 function make text input ($name, $label) { 

5 

16 // 打印 段落 标签 和 label 标 签 : 

17 print '<p><label>' . Slabel . ': '; 

18 

19 // 打印 文本 框 : 

20 print '<input type="text" name="' . S$name . '" size="20" '，; 
21 

22 // 加 入 文本 框 预 设 值 : 

23 if (isset(S$_POST[Sname])) { 

24 print ' value="' . htmlspecialchars($ POST[S$Sname]) . '"'; 
25 } 
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27 // 完成 文本 框 ， 关 闭 Labe1 和 段落 标签 : 
28 print ' /></label></p>'; 


29 

30 } // 结束 make_text_jinput () 函数 。 

31 

32 // 创建 表单 : 

33 print '<form action="" method="post">'; 
34 


35 // 创建 文本 框 : 

36 make text input('first name', 'First Name'); 
37 make text input('last name', 'Last Name'); 
38 make text input('email', 'Email Address'); 


40 print '<input type="submit" name="submit" value="Register!" /></form>'; 


42 ?> 
43 </body> 
44 </html> 


(2) 开始 PHP 代 码 部 分 : 

<?php // 脚本 10-2 - sticky1.php 

(3) 开始 定义 函数 : 

function make_ text_input ($name, $label) { 

make_text_input () 冰 数 有 两 个 参数 ， 两 个 参数 分 别 赋值 给 $name 和 $1label 变 量 。 
(4) 打印 段落 标签 和 1abel 标 签 : 


print '<p><label>' . S$label . 

这 个 函数 生成 的 代码 与 第 8 章 给 出 的 差不多 ， 但 是 这 里 会 将 代码 包括 在 段落 标签 中 ， 文 本 框 
标签 会 放 在 1abel 标 签 中 。 调 用 国 数 时 ， 会 将 标签 的 值 (如 ，First Name) 传递 到 函数 中 。 

(5) 打印 文本 框 : 


print '<input type="text"name="' . $name . '" size="20" ，' 

PHP 的 print 语 句 用 来 创建 HTML input 标 签 ， 但 标签 name 属 性 的 值 来 自 变 量 sname (调用 
国 数 时 赋值 ， 例 如 : first_name) 。 

(6) 如 果 需 要 ， 加 入 文本 框 预 设 值 : 


if (Isset(S$_POST [Sname])) { 
print ' Value="' . 
vhtmlspecialchars($_POST [Sname]) . '" 
} 


第 5 步 中 的 代码 并 没有 结束 文本 框 创建 (没有 使 用 /> 关闭 )， 因 此 还 可 以 加 入 类 似 value= 
"whatever" 的 语句 , 但 这 条 语句 只 能 在 设置 了 $_POST['input_name'] 的 情况 下 才 可 加 入 , 所 
以 这 里 加 入 了 条 件 语句 。 同 第 8 章 的 示例 代码 一 样 ， 元 素 的 value 属 性 值 只 能 在 运行 完毕 
htmlspecialchars() 国 数 时 才 可 以 打印 。 

(7) 完成 文本 框 ， 关 闭 1abel 和 段落 标签 ， 结 束 函数 : 





出 
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print ' /></label></p>'; 
} // End of make text_ input()function. 


(8) 创建 form 标 签 ， 调 用 函数 : 


print '<form action=""method="post">'; 
make_ text_ input('first name', 'First Name'); 


表单 必须 使 用 POST 方法 ， 因 为 函数 检查 的 是 $S_PosT 变 量 的 值 ， 这 点 很 重要 。 如 果 表 单 会 自 
动 提交 回 相同 的 页 面 ， 可 以 忽略 action 值 。 














(9) 创建 另外 两 个 文本 框 : 
make text_ input('last name', 'Last Name'); 
make text_ input('email', 'Email Address'); 


注意 ， 脚 本 以 三 种 不 同 的 方式 使 用 了 同一 个 国 数 三 次 ， 生 成 三 个 完全 不 同 的 文本 框 。 
(10) 完成 表单 : 

print '<input type="submit" name="submit" value="Register!" /></form>'; 
表单 需要 一 个 提交 按钮 ， 测 试 表单 的 粘性 功能 。 

(11) 结束 PHP 和 HTML: 





?> 
</body> 
</html> 


(12) 将 文件 保存 为 sticky1 .php， 放置 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 ， 并 且 在 Web 浏 
览 器 上 测试 (参见 图 10-6 和 图 10-7)。 











First Name: Peter 





Last Name: O'Toole 





Email Address: me@example.com 











( Register! ) 

















图 10-7 表单 元 素 显示 用 户 之 前 输入 的 值 








w 提示 

口 可 以 定义 任何 数量 的 函数 ， 而 不 用 像 本 章 示例 中 描述 的 那样 ， 即 每 个 脚本 中 只 有 一 个 函 

数 。 

口 函数 能 够 接受 的 参数 没有 数量 限制 。 

口 一 旦 已 经 像 这 样 定义 了 自己 的 函数 ， 就 可 以 将 它们 放置 在 外 部 文件 中 ， 并 且 可 以 在 需要 
访问 这 个 函数 的 时 候 请 求 该 文件 。 








10.3 UUUDO0OO0ODOOD 253 





10.3 ”设置 参数 默认 值 
PHP 人 允许 函数 拥有 默认 的 参数 值 。 要 做 到 这 些 ， 只 用 在 函数 定义 中 为 参数 赋值 即 可 .: 





function greeting($who = 'world') { 

print "<p>Hello, Swho!</p>"; Hello, world! 
} 
这 个 函数 将 使 用 预 设 值 ， 除 非 它 获取 到 一 个 覆盖 默认 值 的 Hello, Zoe! 











值 。 换 句 话 说 ， 通 过 为 一 个 参数 设 定 默认 值 ， 可 以 在 调用 函数 


时 旦 现 特定 的 可 选 参数 。 如 果 希 望 为 参数 假定 一 个 特定 的 值 ， 。 图 :0 导 几 站 直 上 没有 任何 
参数 的 函数 (第 一 个 问 




















那么 需要 为 其 设 定 默 认 值 ， 但 仍然 允许 传递 其 他 值 (参见 图 人 

10-8)。 参数 的 函数 意味 着 该 值 
greeting(); 将 会 被 使 用 (第 二 个 问 
greeting('Zoe'); 候 ) 


注意 ,这 不 是 用 户 自 定义 函数 (或 一 个 默认 的 参数 值 ) 的 好 例子 ， 示 例 中 使 用 它 只 是 为 了 方 
便 理解 。 

应 当 总 是 在 其 他 标准 参数 (它们 没有 默认 值 ) 之 后 编写 默认 参数 。 这 是 因为 PHP 按 照 从 调用 
行 获 取 的 顺序 直接 为 参数 赋值 。 因 此 , 不 可 能 出 现 省 略 第 一 个 参数 的 值 但 却 包含 第 二 个 参数 的 情 
况 。 例 如 ,假设 此 时 有 : 


function calculate totall($qty, S$price = 20.00, S$tax = 0.06) {... 


如 采用 代码 行 调用 函数 : 

calculate_total(3, 0.07); 
我 们 期 望 将 $qty 设 置 为 3， 将 Sprice 的 值 保持 为 20 .00， 并 且 将 $tax 的 值 更 改 为 0.07， 但 这 
是 有 问题 的 。 

最 终 的 结果 将 是 Sqty 被 设置 为 3，$price 为 0.07， 而 Stax 则 保持 为 0.06 (参见 图 10-9)， 
这 并 不 是 所 希望 的 输出 结果 。 为 了 达到 期 望 的 结果 ， 正 确 的 方法 是 使 用 下 面 的 代码 : 

calculate_total(3, 20.00, 0.07); 

让 我 们 重新 运行 make_text_input () 国 数 以 获取 对 设 定 参 数 默认 值 的 完整 认识 。 
































function calculate ST ds pc = 20.00, S$tax = 0.06) { 
// Make the calcvlations. 


} 





calculate total (3, 0.07); 





| 











10-9 ”根据 函数 参数 赋值 的 方式 ， 在 调用 函数 时 不 能 “ 跳 过 ” 某 个 参数 











这 编写 一 个 使 用 默认 值 的 函数 











(1) 如 果 sticky1 .php 不 在 开启 状态 的 话 , 在 文本 编辑 器 或 者 IDE 中 打开 它 (参看 脚本 10-3)。 
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(2) 为 函数 make_text_input () 加 入 第 三 个 参数 及 基 


function make text_ input ($name, S$label, 


尽管 我 喜欢 整洁 的 页 面 ， 习 惯 将 所 有 的 文本 
常 比 名 字 要 长 , 因此 最 好 可 以 调整 文本 框 的 大 小 。 











ssize 





其 默 认 值 ， 


20) { 


匡 做 成 同样 的 大 小 , 但 人 的 姓氏 ”和 Emial 地 址 通 
将 文本 框 的 大 小 作为 参数 , 可 以 解决 这 个 问题 。 




















文本 框 大 小 要 有 一 个 默认 值 ， 使 其 成 为 一 个 可 选 的 参数 。 如 果 传 人 第 三 个 参数 ，$size 的 默认 值 


就 会 被 这 个 参数 的 值 取代 。 


脚本 10-3 ”函数 将 接受 两 个 参数 ， 但 其 中 只 有 一 个 是 必需 的 。 如 果 $size 的 值 没有 发 送 给 函数 ， 








那么 它 的 值 将 是 20 
1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 
5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>Sticky Text Inputs</title> 
7 </head> 
8 <body> 
9 <?php // 脚本 10-3 - sticky2.php 
10 /* 脚本 定义 并 调用 函数 ， 该 函数 创建 粘性 文本 框 。*/ 
11 
12 // 函数 创建 粘性 文本 框 。 
13 // 函数 接受 两 个 参数 。 
14 // 第 三 个 参数 是 可 选 的 (包含 默认 值 ) 。 
15 function make text input ($name, $label, $size = 20) { 
16 
17 // 打印 段落 标签 和 label 标 签 : 
18 print '<p><label>' . $label . ': '; 
19 
20 // 打印 文本 框 : 
21 print '‘'<input type="text" name="' . $name . '" size="' . $size . '" '} 
区 
23 // 加 入 文本 框 预 设 值 : 
24 if (isset($_ POST[$Sname])) { 
2.5 print ' value="' . htmlspecialchars ($_POST[S$namel]) me 
26 } 
27 
28 // 完成 文本 框 ， 关 闭 label 和 段落 标签 : 
29 print ' /></label></p>'; 
30 
31 } // 结束 make_text_input () 函数 。 
32 
33 // 创建 表单 : 
34 print '<form action="" method="post">'; 
35 
36 // 创建 文本 框 : 
37 make text input('first name', 'First Name'); 
38 make text input('last name', 'Last Name', 30); 
@ 这 里 指 西 方 人 的 姓氏 。 一 一 编者 注 
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[le 口 忆 


39 make text input('email', 'Email Address', 50); 
41 print '<input type="submit" name="submit" value="Register!" /></form>'; 


43 ?> 
44 </body> 
45 </html> 


(3) 改变 文本 框 创建 语句 ， 加 入 $size 变 量 : 


print '<input type="text"name="' .Sname . '"size="' . $size . '" 
(4) 修改 函数 调用 语句 ， 改 变 文本 框 的 大 小 : 


make text input('first name', 'First Name'); 
make text_ input('last name', 'Last Name', 30); 
make text input('email', 'Email Address', 50); 


现在 ， 第 一 个 文本 框 会 使 用 默认 大 小 ， 而 其 他 的 文本 框 会 更 长 一 些 。 
(5) 将 脚本 保存 为 sticky2 .php， 放 置 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 ， 并 且 在 Web 浏 





览 器 上 测试 (参见 图 10-10)。 


10 








First Name: Peter 





Last Name: O'Toole 





Email Address: me@example.com 








( Register! ) 








图 10-10 现在 函数 可 以 根据 传 入 的 参数 改变 大 小 ， 如 果 不 传 入 参数 ,将 使 用 
默认 大 小 (第 一 个 文本 框 ) 
提示 
口 可 以 使 用 一 个 空 字符 捉 (，') 或 者 worL (不 带 有 引号 ) 为 函数 的 特殊 参数 传递 空 值 。 用 
其 中 任何 一 种 方式 都 将 覆盖 原 有 的 默认 值 。 
D 正如 第 ! 章 中 提 到 的 ,PHP 手 册 用 方 括号 来 标记 函数 参数 . 合 如 , 当 售 用 mumber_tomac0 加 通 汪 
函数 时 ， 舍 入 到 小 数 的 位 数 是 可 选 的 ; 


string number_format (float number [, int decimals]) 











.4 创建 和 使 用 带 有 返回 值 的 函数 


函数 不 仅 可 以 接受 参数 ， 还 能 够 返回 值 ,这 只 需要 两 个 步 又 即 可 。 首 先 ， 在 函数 中 使 用 返回 

















语句 。 第 二 , 在 调用 函数 的 时 候 以 某 种 方式 使 用 输出 。 通 常情 况 下 ,将 会 把 返回 的 值 赋 给 一 个 变 


， 但 是 还 可 以 做 更 多 操作 ， 如 直接 打印 出 输出 结果 。 这 是 接受 两 个 参数 并 返回 一 个 值 的 函数 的 


基本 格式 ， 


256 “0100 0000 





function make_ full name ($first, S$last) { 
Sname = $first . ' ' . $last; 
return S$name; 


} 

可 以 这 样 使 用 函数 : 

$full name = make full name ($fn, $1n); 

这 里 函数 返回 的 值 被 赋 给 一 个 变量 。 它 被 立即 打印 出 来 : 

print make_full name ($fn, $1ln) 

为 了 更 好 地 诠释 这 个 概念 ,我 们 创建 一 个 函数 ， 它 将 执行 一 个 简单 的 计算 并 且 对 结果 进行 格 
式 化 。 这 段 脚 本 将 显示 一 个 用 户 用 ne (参见 图 10-11)。 当 表单 被 提 
交 时 〈 回 到 这 个 相同 的 页 面 )， 将 打印 出 一 个 总 值 (参见 图 10-12)。 












































Your total comes to $572.39. 
Quantity: 221 Quantity: 
Price: 2.59 Price 
( Calculate! ) ( Calculate! ) 
图 10-11 这 个 简单 的 表单 带 有 两 个 需要 计算 的 值 10-12 用户 自 定义 函数 计算 的 结果 

















党 创建 和 使 用 带 有 返回 值 的 函数 


(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 PHP 文 档 ， 命 名 为 calculator .php (参看 脚本 
10-4): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Cost Calculator</title> 

</head> 

<body> 











二 

















脚本 10-4 该 脚本 显示 并 处 理 了 一 个 HIML 表 单 以 便 执行 一 些 基本 的 计算 。 脚 本 使 用 了 一 个 接 
受 两 个 参数 和 一 个 返回 值 的 函数 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Cost Calculator</title> 

</head> 


OUwW 必 wm 
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8 <body> 
9 <?php // 脚本 10-4 - calculator.php 


























10 /* 脚本 显示 并 处 理 HTML 表 单 。 

11 函数 将 数量 乘 以 单价 ， 以 计算 总 值 。*/ 

12 

13 // 函数 执行 计算 功能 。 

14 function calculate total (S$Sdquantity，Sprice) { 

BN 

16 Stotal = $Squantity * S$price; // 计算 。 

17 $total = number_format (S$total，2); // 格式 化 数字 ,保留 两 位 小 数 。 

18 

19 return S$total; // 返回 值 。 

之 0 

21 } // 结束 函数 。 

22 

23 // 检查 表单 是 否 提交 : 

24 if ($_SERVER['REQUEST METHOD'] == 'POST') { 

长 与 

26 // 检查 表单 数据 : 

27 if ( is_ numeric($_POST['quantity']) AND is_ numeric($_POST['price']) ) { 

28 

29 // 调用 函数 并 打印 结果 : 

30 $total = calculate total($ POST['quantity'], $_ POST['price']); 

3 print "<p>Your total comes to $<span style=\"font-weight: 
bold;\">$total</span>.</p>"; 

32 

33 } else { // 输入 的 数值 不 适宜 。 

34 print '<p style="color: red;">Please enter a valid quantity and price!</p>'; 

35 } 

36 

37 } 

38 ?> 

39 <form action="calculator.php" method="post"> 

40 <p>Quantity: <input type="text" name="quantity" size="3" /></p> 

41 <p>Price: <input type="text" name="price" size="5" /></p> 

42 <input type="submit" name="submit" value="Calculate!" /> 

43 </form> 

44 </body> 

45 </html> 

(2) 开始 PHP 代 码 部 分 : 

<?php // 脚本 10-4 - calculator.php 

(3) 定义 函数 : 


function calculate_total (S$Sduantity，Sprice) { 
Stotal = S$Squantity * S$price; 
Stotal = number_ format (S$total, 2); 
return S$total; 


} 
该 函数 接受 两 个 参数 : 数量 和 单价 ， 并且 将 它们 进行 乘法 运算 来 创建 总 价 。 总 价 在 被 函数 返 
回 之 前 会 被 格式 化 。 
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尽管 这 样 使 用 函数 看 上 去 显得 有 点 笨拙 ， 但 是 这 里 将 一 个 单 步 运 算 放 在 函数 中 具有 双重 好 
处 : 首先 ,在 脚本 的 开头 处 放置 这 个 计算 而 不 是 将 它 隐 藏 在 代码 的 其 他 地 方 ， 这 会 使 日 后 在 图 数 
中 查找 和 修改 这 个 计算 时 变 得 更 加 容易 ， 其 次， 可 以 不 用 复制 代码 而 在 脚本 中 重复 这 个 动作 。 

(4) 开始 条 件 部 分 以 查看 是 否 表单 被 提交 : 

if ($_SERVER['REQUEST METHOD'] == 'POST') { 

因为 该 页 面 显示 并 处 理 了 HTML 表 单 ， 所 以 它 用 一 个 条 件 来 检验 页 面 的 请 求 方式 。 如 果 是 
POST 请 求 ， 就 表明 表单 已 经 被 提交 。 

(5) 验证 表单 数据 并 使 用 函数 : 


If ( is numeric($_ POST['gquantity']) AND is numeric($ POST['price']) ) { 
Stotal = calculate total ($_POST['quantity'], $_POST['price']); 
print "<p>Your total comes to $<span style=\"font-weight: 
bold; \">$total</span>.</p>"; 


这 部 分 PHP 代 码 (处 理 被 提交 的 表单 ) 首先 检验 是 否 已 经 填写 数量 和 单价 。 如 果 是 ， 总 价 的 值 
将 通过 调用 calculate_total () 函数 计算 ， 并 将 结果 赋 给 $total 变 量 。 然 后 会 打印 出 结果 。 
(6) 完成 条 件 语 句 : 


} else { 
print '<p style="color: red;">Please enter a Valid quantity and price!</p>'; 


} 















































} 
如 果 某 个 表单 变量 没有 被 正确 地 提交 , 将 会 打印 出 一 条 消息 来 指明 这 个 问题 。 最 后 的 花 括号 
结束 表单 的 提交 条 件 语 句 。 
应 用 一 小 段 CSS 代 码 来 打印 消息 (在 这 里 和 第 (5) 步 中 )。 
(0) 显示 HTML 表 单 : 


?% 

<form action="calculator.php" method="post"> 
<p>Quantity: <input type="text" name="quantity" size="3" /></p> 
<p>Price: <input type="text" name="price" size="5" /></p> 
<input type="submit" name="submit" value="Calculate!" /> 

</form> 


表单 本 身 很 简单 ， 它 向 用 户 请 求 两 个 不 同 的 值 (参见 图 10-8)。 隐 藏 的 表单 输入 作为 处 理 代 
码 的 触发 器 ,表明 表单 已 提交 。 因 为 这 个 表单 是 在 主 提交 条 件 之 外 创建 的 ， 因此 它 总 是 显示 在 页 
面 上 。 











(8) 完成 HTML 页 面 : 
</body> 
</html> 


(9) 保 存 页 面 为 calculator .php, 放置 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 , 并 且 在 Web 
浏览 器 中 进行 测试 (参见 图 10-12)。 
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返回 多 个 值 
ggg 
UUUU0O0O00000 


function some_ function($al, $a2) { 
// 做 你 想 做 的 事情 。 
evn av I LSY 已 


} 
国生 


list($varl, S$var2) = Some_function(Sp1,Sp2) ; 


D0000000 Syi0UU0DPHPIUUU $varid UOUUD $y20 0 0 0 Svar20 


Vv 提示 

口 可 以 在 一 个 函数 中 只 执行 一 条 返回 语句 ， 但 是 相同 的 函数 能 够 有 多 条 返回 语句 。 作 为 示 
例 ， 可 以 编写 一 个 函数 用 来 检验 条 件 并 用 以 指明 条 件 是 否 被 满足 。 在 这 种 情况 下 ， 可 以 
在 函数 中 编写 这 样 的 代码 : 


if (condition) { 
return TRUE; 
} else { 
return FALSE; 
} 


国 数 返回 的 结果 不 是 TRUE 就 是 FALSE， 用 以 指示 是 否 满足 声明 的 条 件 。 


10.5 理解 变量 作用 域 


在 没有 学 习 函 数 之 前 介绍 变量 ,作用 域 没 有 任何 意义 。 因 此 ， 之 前 我 们 并 没有 介绍 变量 作用 
域 。 现 在 , 已 经 对 函数 有 所 了 解 ， 本 节 就 将 再 次 回顾 关于 变量 的 话题 ,并 且 讨 论 变 量 和 函数 是 如 
何 协 同 工 作 的 一 些 细节 。 

正如 在 10.2 节 中 看 到 的 那样 ， 可 以 用 将 变量 作为 参数 的 方式 向 函数 发 送 变 量 。 然 而 ， 也 可 以 
引用 使 用 了 全 局 声明 函数 中 定义 的 外 部 变量 。 正 因为 有 变量 的 作用 域 ， 才 使 这 种 应 用 成 为 可 能 。 
变量 的 作用 域 是 它 存 在 的 领域 。 默认 情况 下 , 脚本 中 编写 的 变量 存在 于 脚本 的 生命 周期 中 。 相反 ，， 
诸如 $_SERVER[' PHP_SELF' ] 的 环境 变量 将 会 一 直 在 服务 器 中 存在 。 

尽管 函数 可 以 创建 一 个 新 级 别 的 作用 域 , 但 函数 变量 (函数 的 参数 如 同 在 函数 中 定义 的 任何 
变量 一 样 ) 只 在 那个 函数 中 存在 ， 并 且 不 能 从 该 函数 之 外 访问 (这 就 是 说 ， 它 们 是 局 部 变量 , 拥 
有 局 部 作用 域 )。 同 样 地 ， 上 默认 情况 下 ， 函 数 外 的 变量 不 能 在 函数 内 使 用 (参见 图 10-13)。 即 使 
某 个 变量 是 函数 调用 的 参数 ， 也 只 是 这 个 变量 的 0 传 给 函数 的 ， 而 不 是 变量 本 身 的 值 。 

global 声 明 大 致 的 意思 是 : 希望 这 个 函数 内 的 变量 能 够 指向 函数 外 具有 相同 名 称 的 变量 。 
换 名 话说 ，global 声 明 将 一 个 拥有 局 部 作用 域 的 局 部 变量 转变 为 拥有 全 局 作用 域 的 全 局 变量 。 
当 变 量 在 函数 之 外 时 (也 就 是 说 假设 函数 被 调用 )， 在 函数 中 对 这 个 变量 所 作 的 任何 变更 都 会 被 
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传递 给 变量 ， 而 不 必 使 用 返回 命令 。 


在 函 

















<?php 


// Whatever code. 






function blah($al, $a2) { 
// Function code. 


} 局 部 作用 域 


全 局 作用 域 





// Whatever code. 
?> 











图 10-13 ”在 脚本 中 加 入 函数 定义 ， 就 添加 了 另 一 个 变量 作用 域 


global 声 明 的 语法 如 下 : 


function function name(s$sargs) { 
global Ss$variable; 
statement(s); 








} 
这 将 导致 另外 一 个 关于 函数 和 变量 的 问题 : 因为 有 变量 作用 域 存在 ,函数 中 的 局 部 变量 是 刁 
数 之 外 的 变量 不 同 的 实体 (也许 有 不 同 的 值 )， 即 使 这 两 个 变量 使 用 完全 相同 的 名 称 。 让 我 





们 进一步 明确 这 一 点 …… 


假设 有 : 

function test($arg) { 
// 做 你 想 做 的 事情 。 

} 

Svar = 1; 

test ($var); 


当 函 数 被 调用 时 ，$var 的 值 将 被 赋 给 $arg， 因 此 它们 的 值 是 一 样 的 , 但 是 它们 的 名 称 不 同 ， 





并 且 是 不 同 的 变量 。 但 是 ， 如 果 函 数 中 的 参数 名 称 也 是 $var 时 : 


function test(Svar) { 
// 做 你 想 做 的 事情 。 

} 

Svar’ = 1; 

test ($var); 


这 样 函数 中 的 $var 变 量 会 被 赋予 同 函数 外 原始 $var 相 同 的 值 ， 但 是 它们 仍然 是 两 个 单独 的 


变量 。 一 个 是 在 函数 内 拥有 作用 域 , 而 另外 一 个 的 作用 域 是 在 函数 外 。 这 意味 着 在 函数 内 和 函数 
外 使 用 名 称 完全 相同 的 变量 不 会 产生 冲突 。 只 是 需要 记 住 ,它们 是 不 同 的 变量 即 可 。 在 函数 内 变 


量 的 











值 只 对 函数 内 的 变量 产生 作用 ， 示 例如 下 (参见 图 10-14)。 
function adqd one($n) { 

S$Sn++; 

print 'Added one!<br/>'; 


10.5 UUUOO0OO0D0OD0 261 





sn = 1; 

print "\$n equals $n<br/>"; 
adqd_one ($n); 

print "\$n equals S$n<br/>"; 


这 些 都 是 正确 的 ， 除 非 使 用 全 局 声明 ， 当 然 ， 这 将 使 这 两 个 变量 变 得 相同 (参见 图 10-15): 


function add one() { 
global Sn; // 相同 ! 





Sn++; 

print 'Added one!<br/>'; 
} 
和 二， 江汉 
print "NSn equals S$n<br/>"; 
add_one(); 
print "\$n equals S$n<br/>"; 


注意 ， 在 这 个 例子 中 ， 不 再 需要 将 变量 的 值 传 递 到 函数 中 。 








Sn equals 1 Sn equals ] 
Added one! Added one! 


Sn equals 1 Sn equals 2 


























WA 


10-14 改变 函数 内 局 部 变量 的 值 图 10-15 改变 函数 内 的 全 局 变量 
不 会 影响 同名 的 全 局 变量 将 会 影响 函数 外 的 变量 


为 了 诠释 变量 的 作用 域 ， 我 们 用 global 声 明 重 写 calculator .php 肢 本。 


过 使 用 global 声 明 


(1) 如 果 calculator .php 不 是 开启 状态 的 话 ， 在 文本 编辑 器 或 者 IDE 中 打开 它 (参看 脚本 
10-4) 。 
(C) 在 函数 定义 之 前 ， 添 加 下 面 的 代码 (参看 脚本 10-5): 


stax =- 8.75; 





























脚本 10-5 ”脚本 中 的 函数 能 够 使 用 $tax 变 量 (即便 它 并 没有 被 传递 给 函数 ) ， 这 要 归功 于 0 
global 声 明 








<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> 
<title>Cost Calculator</title> 

</head> 

<body> 

<?php // 脚本 10-5 - calculator.php #2 

/* 脚本 显示 并 处 理 HTML 表 单 。 

哆 数 将 计算 包含 税率 在 内 的 总 值 。 */ 


FADo、eO WwWN 


© 
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13 // 定义 税率 : 
14 S$tax = 8.75; 


16 // 函数 执行 计算 功能 。 
17 function calculate total (S$Sdquantity，Sprice) { 


18 

19 global S$tax; 

20 

2 Stotal = Squantity * Sprice; // 计算 。 
22 $taxrate = ($tax / 100) + 1; 


23 $total = $total * S$taxrate; // 纳入 税率 后 的 总 值 。 
24 Stotal = number_format (Stotal，2); // 格式 化 数字 ， 保 留 两 位 小 数 。 





25 

26 return Stotal; // 返回 值 。 

号 汪 

28 } // 结束 函数 。 

29 

30 /1/ 检查 表单 是 否 提交 : 

31 if (isset(S$_POST['submittedq'])) { 

32 

33 // 检查 表单 数据 : 

34 if ( is numeric($_ POST['gquantity']) AND is numeric($ POST['price']) ) { 

35 

36 // 调用 函数 并 打印 结果 : 

3:7 Stotal = calculate total($_POST['gquantity'], $_POST['price']); 

38 print "<p>Your total comes to $<span style=\"font-weight: bold;\">$total 
</span>, including the S$tax percent tax rate.</p>"; 

39 

40 } else { // 输入 的 数值 不 适宜 。 

41 print '<p style="color: red;">Please enter a valid aquantity and price!</p>'; 

42 } 

43 

44 } 

45 ?> 

46 <form action="calculator.php" method="post"> 

47 <p>Quantity: <input type="text" name="quantity" size="3" /></p> 

48 <p>Price: <input type="text" name="price" size="5" /></p> 

49 <input type="submit" name="submit" value="Calculate!" /> 

50 <input type="hidden" name="submitted" value="true" /> 

51 “</form> 

52 </body> 


53 </html> 

创建 一 个 $tax 变 量 ， 使 它 拥 有 将 在 费用 计算 中 用 到 的 一 系列 的 值 。 它 被 赋予 函数 之 外 的 一 
个 值 ， 这 是 因为 在 这 之 后 它 将 会 在 脚本 的 主体 部 分 被 用 到 。 

(3) 在 函数 定义 部 分 ， 添 加 一 个 全 局 声明 

global S$tax; 

这 个 声明 告诉 函数 要 将 相同 的 $Stax 变 量 作 为 存在 于 函数 之 外 的 变量 。 

(4) 在 国 数 中 的 Stotal 被 格式 化 之 前 ， 重 新 用 税率 计算 值 
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staxrate = (Stax / 100) + 1; 
Stotal = S$total * S$taxrate; 


为 了 将 税 值 加 到 总 值 中 , 需要 先 将 税 除 以 100， 以 获得 一 个 百分比 。 然 后 为 这 个 值 加 上 1 以 获 
得 一 个 乘 数 。 将 这 个 结果 同 先前 的 总 值 相 乘 以 获得 新 的 总 值 ， 即 为 最 后 的 总 值 。 

请 注意 ， 是 用 一 个 Staxrate 变 量 (基于 $tax) 来 执行 这 个 计算 的 。 这 是 因为 在 这 之 后 将 打 
印 出 $tax 的 值 ， 并 且 在 这 里 对 之 所 作 的 任何 更 改 都 能 反映 出 来 《因为 它 是 一 个 全 局 变量 )。 

(5) 在 主 print 代 码 行 ( 在 函数 调用 之 后 ) 添加 下 面 的 代码 以 便 打印 出 税率 : 


print "<p>Your total comes to $<span style=\"font-weight: 
"bold;\">$total</span>, including the S$tax percent tax rate.</p>"; 


在 脚本 的 开头 部 分 被 定义 的 变量 $tax 最 终 在 结尾 处 被 打印 。 如 果 还 没有 使 用 函数 内 的 
$taxrate 变 量 ， 并 且 替 换 为 Stax 这 个 全 局 变量 ， 对 计算 结果 的 影响 将 通过 打印 值 的 方式 表现 
出 来 。 

(6) 保存 脚本 ,放置 在 启用 了 PHP 服 务 器 上 适当 的 目录 中 ,并 在 Web 浏 览 器 中 进行 测试 (参见 
图 10-16 和 图 10-17)。 










































































Your total comes to $2,114.59, including the 8.75 percent tax rate. 
Quantity: [15 | Quantity: [| 
Price: |129.63 | Price: 
Calculater) CE 
图 10-16 再 次 运行 表单 图 10-17 计算 过 程 现 在 使 用 了 $tax 这 个 全 局 变量 
w 提示 


口 常量 和 超 全 局 数组 ($_GET、$_POST、$_COOKIE 和 $_SESSION) 能 带 来 额外 的 益处 ， 
它们 通常 不 需 全 局 声明 就 能 在 国 数 内 使 用 〈 正 因 如 此 ， 它 们 被 称 为 超 全 局 )。 
口 每 个 函数 都 有 它 自己 的 、 独 立 的 局 部 作用 域 。 








函数 设计 理论 
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10.6 回顾 和 实践 


IE 


国 


如 果 你 对 本 市 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ,网 站 地 址 : www.LarryUllman. 





com/forum/。 


10.6.1 回顾 


口 用 户 自 定义 函数 的 基本 语法 是 什么 ? 

口 用 户 自 定义 函数 的 命名 规则 是 什么 ? 

口 用 户 自 定义 函数 的 参数 命名 规则 是 什么 ? 

口 如 何 为 自 定义 函数 设置 参数 默认 值 ? 

口 在 10.5 季 的 示例 代码 中 要 使 用 \$n? 如 果 不 加 反 斜 杠 会 发 生 什么 ? 
口 什么 是 变量 作用 域 ? 函数 中 参数 变量 的 作用 域 是 什么 ? 

口 包含 文件 中 变量 作用 域 是 什么 ? 注意: 这 个 问题 比较 难 ! 








10.6.2 ”实践 


口 设置 menus .php 中 的 函数 带 有 两 个 参数 ， 一 个 参数 表示 起 始 年 ， 男 一 个 参数 





口 重 写 make_text_input () 函数 ， 以 便 可 以 告诉 它 是 在 $_POST 中 还 是 在 $_G] 
存在 的 值 。 

口 在 make_text_input () 函数 基础 上 创建 一 个 新 函数 , 根据 函数 调用 方式 , 创 
框 和 一 个 密码 框 。 

口 自己 设计 一 个 脚本 ， 脚 本 中 要 使 用 自 定义 函数 。 











表示 生成 的 


总 年 数 。 第 二 个 参数 有 一 个 默认 值 。 重 写 国 数 体 ， 在 year 的 for 循 环 中 使 用 这 些 参数 值 。 


ET 中 查询 已 


建 一 个 文本 


文件 和 目录 








本 章 内 容 

口 文件 权限 

口 写 入 文件 

D 锁定 文件 

口 读 取 文 件 

口 处 理 文件 上 传 
口 导航 目录 

口 创建 目录 

口 增 量 读 取 文件 
口 回顾 和 实践 





要 使 Web 应 用 程序 真正 做 到 更 上 一 层 楼 ， 需 要 掌握 存储 和 检索 数据 的 方法 。 使 用 PHP 有 两 种 
主要 的 用 于 存储 数据 的 方法 : 使 用 文件 (和 目录 ) 或 数据 库 。 本 章 讨论 前 者 ， 而 第 12 章 讨论 后 者 。 
理解 这 两 种 方法 值得 花 一 些 时 间 。 尽 管 数 据 库 可 以 比 基 于 文件 的 系统 更 强大 也 更 安全 , 但 是 通过 
读 写 服 务 器 上 的 简单 文本 文件 来 发 送 和 检索 信息 ， 能 够 做 到 的 事情 也 会 令 人 很 惊讶 ! 

本 章 将 学 到 有 关 文 件 权 限 的 知识 ， 然 后 了 解 写 入 、 读 取 和 锁定 文件 。 在 这 之 后 ， 你 会 看 到 如 
何 使 用 PHP 处 理 文 件 上 传 、 如 何 创建 目录 ， 以 及 从 文件 中 读 取 数 据 的 另 一 种 方法 。 最 后 的 两 个 示 
例 还 演示 了 一 个 简单 的 基于 文件 的 注册 和 登录 系统 , 你 可 以 在 自己 的 Web 应 用 程序 中 使 用 该 系统 。 


11.1 文件 权限 


在 尝试 写 入 和 读 取 文件 之 前 ,必须 理解 文件 权限 。 这 个 主题 的 范 围 非常 广泛 , 读者 也 许 希 望 能 
够 持续 深入 讨论 , 但 这 里 的 讨论 只 是 一 个 开始 。 首先 本 书 要 说 明 的 是 , 这 里 提 到 的 很 多 信息 只 会 影 
响 非 Windows 用 户 。 根 据 我 的 经 验 ， 当 在 Windows 计 算 机 上 运行 PHP 时 ， 这 些 预备 步骤 并 不 是 必需 
的 。 然 而 ， 从 整体 上 理解 权限 是 件 好 事 ， 尤 其 是 今后 可 能 会 在 非 Windows 服 务 器 上 运行 PHP 脚 本 。 
权限 指定 了 在 一 个 文件 或 目录 上 ， 用户 可 以 做 些 什么 。 其 选项 包括 读 取 、 写 入 和 执行 (实际 
上 , 文件 可 以 指定 为 可 执行 ,而 目录 是 可 搜索 的 )。 在 此 之 上 ， 这些 选项 可 以 分 别 设置 给 3 种 类 型 
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的 用 户 : 

口 文件 的 所 有 者 (创建 文件 或 将 其 放置 到 服务 器 上 的 人 )， 

口 某 特 定 组 的 成 员 ， 由 服务 器 管理 员 设 置 ， 并且 包 括 文件 所 有 者 ，; 
口 其 他 用 户 (不 属于 前 两 个 分 类 的 用 户 )。 

还 有 一 个 隐 仿 的 everyone 级 别 ， 它 包括 前 面 提 到 的 所 有 用 户 。 

举 个 例子 ， 如 果 使 用 FTP 将 文件 上 传 到 服务 器 ， 连 接 服务 器 的 账户 就 是 文件 的 所 有 者 。 文 件 
的 默认 权限 是 所 有 人 都 可 以 读 取 ， 但 只 有 所 有 者 可 以 修改 文件 ( 即 写 入 )。 

问题 是 ， 在 大 多 数 情况 下 ，PHP 脚 本 得 通过 Web 服 务 器 应 用 程序 运行 ， 并 被 视 为 一 个 不 同 的 
服务 器 用 户 。 因 此 ，PHP 和 Web 服 务 器 能 够 读 取 上 传 的 文件 (从 而 让 Web 浏 览 器 可 以 浏览 这 些 文 
件 )， 但 在 默认 情况 下 ，PHP 无 法 修改 这 些 文件 。 

对 于 本 章 中 的 示例 ，PHP 需 要 能 够 对 一 些 文件 和 目录 进行 写 入 。 也 就 是 说 ， 你 必须 知道 如 何 
修改 文件 和 目录 的 权限 。 也 就 是 说 让 文件 或 目录 可 写 (减少 权限 限制 )， 这 可 能 会 带 来 安全 问题 ， 
因此 只 有 在 绝对 必需 的 时 候 才 能 这 样 做 。 

最 后 要 解释 一 个 人 们 普遍 混 靖 的 概念 “用 户 ” 究竟 指 的 是 什么 ?》 用 户 是 在 计算 机 上 创建 
的 账户 。 在 你 自己 的 计算 机 上 ， 可 能 只 有 一 个 用 户 (你 自己 ) 或 几 个 用 户 。 服 务 器 上 一 般 会 有 多 
个 用 户 , 但 是 很 多 用 户 账户 并 不 与 登录 服务 器 的 真实 用 户 相 对 应 , 而 是 与 运行 在 服务 器 上 的 进程 
相对 应 。 例 如 ， 服 务 器 上 可 能 有 一 个 用 户 进程 用 来 处 理 所 有 Web 请 求 ， 还 有 另 一 个 用 户 运行 数据 
库 应 用 程序 。 最 重要 的 是 要 知道 ,“ 用 户 ” 不 是 指使 用 这 台 计 算 机 的 人 , 还 要 知道 everyone 指 的 是 
“服务 器 上 的 everyone”。 如 果 你 在 服务 器 上 创建 了 一 个 对 所 有 用 户 可 写 的 文件 或 目录 ， 并 不 表示 
在 Internet 上 的 所 有 人 都 可 以 写 这 个 文件 或 目录 。 总 之 ， 用 户 必 须 是 服务 器 中 认可 的 账户 。 




































































Web 根 目录 
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C:\inetpub\wwwroot\filename.php[| O000000000000U /var/web/ 
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example.com/filename.php[| http://localhost/filename.php[] 
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加 加 时 加 二 
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11.1.1 创建 文本 文件 


在 第 一 个 示例 中 ， 你 将 操作 服务 器 上 名 为 quotes .txt 的 文本 文件 。 如 果 PHP 脚 本 不 具备 操 
作 该 文件 所 需 的 正确 权限 ， 将 会 看 到 类 似 图 11-1 所 示 的 错误 消息 。 在 继续 进行 本 章 内 容 之 前 ， 应 
该 在 服务 器 上 创建 quotes .txt 并 为 其 建立 权限 。 























Warning: fopen(./guotes.txt) [function fopen}: failed to open stream: 
Permission denied in 
/Users/larryullman/Sites/phpvqs4/add_quote.php on line 20 


图 11-1 ”尝试 对 一 个 文件 执行 服务 器 所 不 允许 的 操作 会 导致 出 现 “...failed to open stream: 
Permission denied...” 消 息 。 在 这 里 服务 器 禁止 使 用 fopen () 函数 尝试 以 写 入 方式 
来 打开 quotes .txt 


(D 打开 文本 编辑 器 或 IDE 并 新 建 一 个 空白 文档 。 
CO) 不 要 向 文件 中 输入 任何 内 容 ， 将 其 保存 为 quotes .txt。 
(3) 将 文件 移动 到 支持 PHP 的 服务 器 上 的 Web 根 目录 之 外 (参见 图 11-2)。 












































Web 根 目录 的 父 目录 。 ”Web 根 目录 


= 二 一 
i- = 一 (2¥- 一 一 | : 命 add_quote.php 
I 
I 
I 
1 


:3 
= 他 quotes.txt 














WN 








11-2 理想 情况 下 ，quotes .txt 文 件 应 该 放置 在 和 Web 文 档 目 录 同 级 的 目录 中 
(也 就 是 说 ， 不 在 Web 文 档 目 录 之 中 ) 
框 注 “Web 根 目录 ”解释 了 应 该 将 文件 放置 到 相对 于 Web 目 录 的 什么 位 置 ， 并 解释 了 为 什么 
要 这 样 做 。 
w 提示 
口 如 果 可 以 在 服务 器 上 找到 提交 的 文件 名 ， 则 file_exists () 函数 会 返回 TRUE。 这 可 以 
用 于 在 进行 任何 操作 之 前 测试 文件 是 否 存在 。 
if (file exists('somefile.ext')) { ... 
口 假设 PHP 具 备 一 个 目录 上 的 写 权 限 , 则 可 以 使 用 PHP 在 该 目录 中 创建 一 个 空白 文档 。 这 是 
通过 使 用 touch () 函数 来 完成 的 : 


touch('somefile.ext'); 
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11.1.2 ”设置 文件 权限 


前 面 的 步骤 顺序 似乎 看 起 来 有 些 奇 怪 , 但 要 为 一 个 文件 设置 权限 , 该 文件 首先 必须 是 存在 的 。 
不 过 只 需要 一 个 空 的 文件 ， 因 为 稍 后 会 使 用 PHP 向 其 中 写 入 数据 。 

该 示例 的 最 终 预 期 结果 是 为 任何 人 授予 在 quotes .txt 上 的 读 和 写 (但 不 包括 执行 ) 权限 。 
如 何 做 到 这 一 点 取决 于 : 
口 是 在 本 地 计算 机 还 是 在 远程 服务 器 上 运行 PHP， 
口 支持 PHP 的 计算 机 所 使 用 的 操作 系统 。 
遗憾 的 是 , 指出 每 个 用 户 在 各 种 环境 下 设置 权限 的 步 又 是 不 可 能 的 , 但 这 里 给 出 了 一 些 可 用 
的 概略 指导 原则 和 步骤 。 
党 在 远程 服务 器 上 设置 文件 权限 




















口 很 多 ISP 会 为 用 户 提供 一 个 基于 Web 的 控制 面板 ， 可 以 用 于 设置 文件 权限 (参见 图 11-3) 








以 及 其 他 主机 参数 。 
Change permissions for file quotes.txt 
Read mode Write mode Execute/search mode 
Owner 图 加 口 
Group 图 以 口 
Others 图 v 日 
| OK | | Cancel | 

















图 11-3 ”该 控制 面板 由 主机 公司 提供 ， 可 以 用 来 调整 文件 权限 
口 也 可 以 使 用 FTP 客 户 端 来 修改 文件 的 权限 (参见 图 11-4)。 





quotes.txt Info 


@ 
三 quotes.txt 


Kind: Plain Text 
e: 0 B (0 bytes) 


Wed, Dec 29, 2010 10:37 AM 


Execute 














图 11-4 TransmitFTP 应 用 程序 使 用 弹出 窗口 来 设置 文件 权限 
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仿 在 本 地 计算 机 上 设置 文件 权限 











口 如 果 使 用 自己 的 Windows 计 算 机 ， 则 可 能 无 需 修改 权限 。 要 验证 这 一 点 ， 可 以 首先 尝试 一 








下 每 一 个 示例 。 如 果 PHP 不 能 向 文件 或 目录 中 写 入 数据 , 则 需要 使 用 下 面 的 建议 修改 权限 。 
口 Windows 用 户 可 以 通过 查看 文件 或 目录 的 属性 来 修改 权限 。 具体 的 对 话 框 对 于 不 同 版 本 的 
Windows 来 说 是 不 一 样 的 ， 但 需要 调整 的 只 是 谁 可 以 访问 该 文件 以 及 如 何 访问 。 

口 Mac OS X 用 户 必 须 在 Finder 中 选中 文件 ， 并 从 File 菜 单 中 选择 Get Info。 在 这 里 ， 使 用 
Ownership & Permissions 子 面板 来 调整 文件 的 权限 (参见 图 11-5 ) 。 

口 在 Unix 上 (包括 Linux 和 Max OS X 用 户 )， 假 如 具有 相应 的 权限 ， 则 可 以 在 终端 窗口 中 使 
用 chmoqd 0666 quotes .txt 命 令 行 。 














AMA quotes.txt Info 


-~ quotes.txt Zero KB 
Modified: Today 10:30 AM 


= Spotlight Comments: 


bP Ceneral: 








bP More Info: 





Pb Name & Extension: 





P Open with: 





bP Preview: 





vw Sharing & Permissions: 


You can read and write 





Name Privilege 
县 larryullman (Me) $+ Read & Write 
staff $ Read & Write 
I everyone :+ Read & Write 
和 ~ [可 > os 














图 11-5 Mac OS X 的 Get Info 面 板 可 以 用 于 调整 文件 的 所 有 权 和 权限 ， 以 及 其 他 一 些 东西 


Vv 提示 

口 很 多 操作 系统 并 没有 PHP 用 户 。PHP 用 户 其 实 就 是 运行 Web 服 务 器 应 用 程序 (如 Apache 或 
IIS) 的 用 户 。 在 Unix 家 族 中 ，Apache 通 常 作为 nobody 运 行 。 在 Windows 上 ，Web 服 务 器 经 
常 以 当前 登录 用 户 身份 运行 (该 用 户 可 能 能 够 创建 文件 )， 这 意味 着 无 需 修改 文件 权限 。 

口 如 果 对 Telnet 和 chmod 很 熟悉 ， 应 该 能 够 理解 数值 0666 的 含义 。 但 这 里 还 是 要 解释 一 下 ， 

因为 有 些 读者 并 不 熟悉 。0 只 是 一 个 前 级 ， 指 出 该 数值 是 以 八进制 格式 书写 的 。 每 个 6 都 

是 由 读 (4) 和 写 〈2) 权限 相 加 得 到 的 一 一 第 一 个 6 赋 给 文件 或 目录 的 所 有 者 ， 然 后 是 组 ， 

最 后 是 其 他 用 户 。 相 对 应 ，0777 人 允许 各 种 类 型 的 用 户 具备 读 (4)、 写 (2) 和 执行 (1) 
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权限 。 该 数字 适用 于 Unix 系 列 的 操作 系统 (Linux、Solaris 和 Mac OS X) 。 

口 PHP 具 有 很 多 用 于 修改 文件 或 目录 权限 的 函数 ,包括 chgrp () 、chown () 和 chmod()。 然 

而 ， 只 有 当 PHP 已 经 具备 了 修改 文件 或 目录 的 权限 后 ， 这 些 国 数 才能 工作 。 

口 与 运行 在 专门 的 服务 器 上 相 比 ， 运 行 在 共享 托管 环境 中 的 Web 网 站 面临 更 多 的 安全 问题 ， 
因为 这 种 环境 中 的 用 户 数量 很 多 。 举 例 来 说 ， 如 果 你 创建 了 一 个 所 有 用 户 都 可 写 的 文件 
或 目录 ， 那 么 所 有 可 以 访问 该 服务 器 的 用 户 都 可 以 修改 它 (前提 是 这 些 用 户 知 道 有 这 么 
一 个 文件 或 目录 ， 而 且 也 没有 其 他 权限 限制 ) 。 


11.2” 写 入 文件 


要 从 文件 中 读 取 内 容 ， 首先 需要 向 其 中 写 入 一 些 内 容 ， 因 此 本 章 首 先 介绍 写 入 文件 。 写 入 文 
件 最 简单 的 方式 是 使 用 file_put_contents () 国 数 : 

file put contents ($file, S$data); 

这 个 函数 是 PHP 5 中 添加 的 ， 它 会 打开 文件 并 写 入 数据 。 第 一 个 参数 是 文件 名 。 可 以 使 用 绝 
对 路 径 ， 也 可 以 使 用 相对 路 符 (参见 框 注 “文件 路 径 ”)。 第 二 个 参数 是 要 写 入 的 数据 ， 可 以 是 字 
符 串 、 数 字 或 数组 〈 只 能 是 一 维 数组 ， 不 能 使 用 多 维 数组 ) 。 无 论 是 文件 还 是 数据 都 不 强制 使 用 
变量 ， 但 使 用 变量 是 常用 的 方法 。 

如 果 文 件 不 存在 ， 函 数 会 尝试 创建 它 。 如 果 文 件 存在 ,文件 的 内 容 会 被 新 的 数据 取代 。 如 果 
需要 将 新 内 容 扎 加 到 文件 中 ， 加 入 FILE_APPEND 常 量 作 为 第 三 个 参数 : 

file put contents ($file, S$data, FILE APPEND); 

当 向 文件 中 追加 数据 时 ， 通 常 需要 每 块 数据 独占 一 行 ， 因 此 每 次 提交 时 都 应 该 以 一 个 运行 
PHP 的 计算 机 操作 系统 的 换行 符 结 束 。 换 行 符 可 以 是 : 
口 \n Unix 和 Mac OS X ; 
口 \r\n Windows。 

这 里 需要 加 上 双 引 号 以 使 转 义 序列 正常 工作 。 

可 以 使 用 PHP 专 用 的 常量 PHP_EOL， 表 示 当 前 操作 系统 的 换行 符 〈 即 ，\n 或 \r\n): 

file_put_contents(Sftile,Sdqata . PHP_EOL, FILE APPEND); 

如 果 使 用 的 不 是 PHP 5 或 以 上 版 本 ， 将 无 法 使 用 file_put_contents () 函数 ， 那 么 你 就 必 
须 使 用 旧 方 法 写 入 文件 首先， 打开 文 件 ， 其 次 ， 写 入 数据 ， 最后， 关闭 文件 。 


$fp = fopen($file, mode); 
fwrite($fp, S$data . PHP_EOL); 
fclose($fp); 


要 写 入 一 个 文件 ， 必须 在 打开 它 时 创建 文件 指针 。PHP 使 用 fopen () 函数 返回 的 文件 指针 来 
引用 这 个 打开 的 文件 。 

打开 文件 时 ， 需 要 重点 考虑 的 是 所 使 用 的 模式 (mode)。 模 式 指 定 了 打开 文件 的 方式 ， 这 取 
决 于 你 希望 对 文件 进行 的 操作 。 最 为 宽松 的 模式 是 a+， 它 允许 读 取 或 写 入 一 个 文件 。 如 果 文 件 不 
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存在 则 会 创建 文件 ， 并 且 自 动 将 新 的 数据 追加 到 文件 的 末尾 (因此 是 a)。 反 之 ，r 只 人 允许 从 文件 
中 读 取 。 表 11-1 列 出 了 所 有 可 能 的 模式 。 每 种 模式 都 可 以 追加 一 个 pb 标志 2 进 制 模式 打 
开 文件 。 对 于 需要 在 多 种 操作 系统 上 读 取 的 文件 来 说 ， 这 是 一 种 较为 安全 的 选择 。 


表 11-1 fopen() 模 式 























































































































模 式 含 义 
r 只 读 ， 从 文件 的 起 始 位 置 开始 读 取 
了 读 或 写 ， 从 文件 的 起 始 位 置 开始 
Ww 只 写 ， 如果 文件 不 存在 则 创建 文件 ， 并 且 复 盖 现 有 内 容 
W+ 读 或 写 ， 如 果 文 件 不 存在 则 创建 文件 ， 并 且 在 写 入 时 覆盖 现 有 内 容 
a 只 写 ; 如 果 文 件 不 存在 则 创建 文件 , 将 新 数据 追加 到 文件 末尾 (保留 任何 现 有 数据 并 向 其 中 添加 新 数据 ) 
at 读 或 写 ， 如 果 文 件 不 存在 则 创建 文件 ， 将 新 数据 追加 到 文件 末尾 〈 写 入 时 ) 
x 只 写 ， 如 果 文 件 不 存在 则 创建 文件 ， 如 果 文 件 存在 则 什么 也 不 做 (并 发 出 一 个 警告 ) 
x+ 读 或 写 ， 如 果 文 件 不 存在 则 创建 文件 ， 如 果 文 件 存在 则 什么 也 不 做 〈 并 在 写 入 时 发 出 一 个 警告 ) 
fwrite() 国 数 根据 选中 的 模式 向 文件 中 写 入 新 数据 (作为 函数 调用 的 第 二 个 参数 进行 传 


递 )。 

写 入 过 程 的 最 后 一 步 ， 是 再 次 使 用 文件 指针 来 调用 fclose () 函数 以 关闭 文件 : 

fclose ($fp); 

er lee. 将 用 户 提交 的 文本 保存 为 纯 文本 文件 〈 参 见 图 11-6)。 在 本 章 稍 
后 的 部 分 中 ， 还 会 创建 其 他 PHP 脚 本 来 检索 并 随机 显示 这 些 文本 。 

ev 还 有 一 个 需要 介绍 的 函数 一 一 is_writable()。 这 个 函数 返回 布尔 
值 ， 表 示 指 定 文件 是 否 可 被 写 入 : 

if (is_ writable($file)) {... 


在 向 文件 (或 目录 ) 中 写 入 数据 之 前 ， 调 用 这 个 函数 ， 避免 权限 错误 。 








Enter your quotation here. 








Add This Quote! 


11-6 这 个 简单 的 表单 允许 用 户 提交 一 段 文本 并 写 人 到 文本 文件 中 





























文件 路 径 


D00000000000000000000U0UD0 absomteld 0 0 UU relatived 
加 
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DC:\somedir\somefile.txt[| Windows[ 

口 /Users/username/somefile.txt[|] Mac OS XI[T]| 

Jo0gogo0o000g0g0gogo0o0geaNg googogooggoggogogogdoo 

口 ElileA.tExtl DD 

OD ./fileA.txt[| DOD 

DO dirB/fileB.txt[| dirB[| [D0 DO 

Lee 0 

oe OD thematic ll 
上 

DU Unixd Linux Mac OS XO 000000000000000000UUUO0U0UUUO 
四 ggg 

本 

国 











> 写 入 一 个 外 部 文件 








(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 PHP 文 档 ， 命 名 为 add_quote .php (参看 脚本 11-1): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Add A Quotation</title> 

</head> 

<body> 


脚本 11-1 该 脚本 用 于 获取 用 户 提交 的 文本 并 将 其 存储 到 文本 文件 中 




















工 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>Add A Quotation</title> 

7 </head> 

8 <body> 

9 ”<?php // 脚本 11-1 - add_quote.php 

10 /* 脚本 显示 并 处 理 HTML 表 单 。 脚 本 从 文本 框 中 获取 文本 并 将 其 存储 到 文本 文件 中 。 */ 

11 

12 // 标识 要 使 用 的 文件 名 : 

13 $file = '../quotes.txt'; 

14 

15 // 检查 表单 是 否 提 交 : 

16 if ($_SERVER['REQUEST METHOD'] == 'POST') { // 处 理 表单 。 

入 县 

18 if ( !empty(S_POST['auote']) && ($_POST['gquote'] != 'Enter your quotation 
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here.') ) { // 检查 输入 的 文本 。 

19 

20 if (is_writable($file)) { // 确认 文件 可 以 写 入 数据 。 

2 

22 file put contents($file, $ POST['quote'] . PHP EOL, FILE APPEND); 

// 写 入 数据 。 

23 

24 // 打印 一 条 消息 : 

5 print '<p>Your quotation has been stored.</p>'; 

26 

27 } else { // 无 法 打开 文件 。 

28 print '<p style="color: red;">Your quotation could not be stored due to a 

SYStem error.</p>'; 

29 } 

30 

31 ) else { // 没有 输入 文本 。 

32 print '<p style="color: red;">Please enter a quotation!</p>'; 

33 } 

34 

35 】 // 结束 提交 条 件 语句 。 

36 

37 ”// 完成 PHP 节 并 显示 表单 : 

38 ?> 

39 

40 <form action="add quote.php" method="post"> 

41 <textarea name="quote" rows="5" cols="30">Enter your quotation here.</textarea> 
<br/> 

42 <input type="submit" name="submit" value="Add This Quote!" /> 

43 </form> 

44 

45 </body> 

46 </html> 

(2) 创建 PHP 代 码 片段 : 

<?php // 脚本 11-1 - add_quote.php 

Stile ev.. /dquotes: txt's 


由 于 这 个 脚本 会 引用 同一 个 文件 两 次 ,因此 最 好 将 这 个 文件 用 一 个 变量 标识 。 这 样 今后 如 果 
需要 改变 文件 名 或 文件 地 址 ， 就 只 需要 修改 一 行 代码 。 

标识 的 文件 名 为 auotes .txt， 这 个 文件 应 该 位 于 该 脚本 的 上 一 级 目录 中 〈 比 如， 在 Web 根 
目录 中 ， 参 见 图 11-2)。 参 见 框 广 “文件 路 径 ” 了 解 更 多 相关 语法 。 

G3) 查看 表单 是 否 已 经 被 提交 

if (S$_SERVER['REQUEST METHOD'] == 'POST') { 


该 页 面 同时 显示 和 处 理 HTML 表 单 。 这 个 条 件 语 句 检 查 表单 是 否 已 经 被 提交 ， 如 果 已 提交 ， 
则 需要 将 文本 写 入 到 文本 文件 中 。 
(4) 检查 输入 的 文本 : 


if ( !empty($_POST['quote']) && ($_POST['quote'] != 'Enter your quotation here.') ) { 
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这 个 简单 的 条 件 语 句 验 证 了 用 户 提供 的 数据 。 第 一 部 分 确保 了 $_POST[' quote'] 变 量 不 为 





空 ， 第 二 部 分 确保 了 该 变量 仍然 没有 默认 值 (如 图 11-6 所 示 )。 

(5) 确认 文件 可 以 写 入 数据 : 

if (is_ writable(s$sfile)) { 

将 条 件 语句 放置 在 一 个 条 件 语 句 中 , 通过 这 种 方式 可 以 令 PHP 脚 本 只 在 文件 能 够 成 功 打 开 时 
尝试 向 其 中 写 入 内 容 。 


(6) 将 数据 写 入 到 文件 ， 关 闭 文 件 ， 然 后 打印 一 条 消息 : 


file put contents($file, $_POST['quote'] . PHP_EOL, FILE_ APPEND); 
print '<p>Your quotation has been stored.</p>'; 


第 一 行 代码 将 用 户 提 交 的 数据 写 入 到 文件 之 中 。 在 写 入 数据 中 加 入 了 PHP_EOL 常 量 , 因此 每 

















次 提交 的 数据 都 会 独占 一 行 。 


消息 ， 


如 果 使 用 的 不 是 PHP 5 或 以 上 版 本 ， 可 以 使 用 以 下 代码 代替 : 


$sfp = fopen ($file, 'ab') 
fwrite($fp, $_POST['gquote'] . PHP_ EOL); 
fclose($fp); 


0) 完成 条 件 语句 : 


} else { // 无 法 打开 文件 。 
print '<p style="color: red;">Your quotation could not be stored due 
一 上 oO a system error.</p>'; 
} 
} else { // 没有 输入 文本 。 
print '<p style="color: red;">Please enter a quotation!</p>'; 





} 
// 结束 提交 条 件 语 向。 


} 
第 一 个 else 检 查 PHP 是 否 打 开 了 用 于 写 入 文件 的 条 件 语句 (参见 图 11-7)。 如 果 看 到 了 这 个 
则 很 可 能 是 由 权限 引发 的 问题 或 文件 引用 不 正确 。 第 二 个 else 完 成 了 条 件 语 句 ， 检 查 文 




















本 是 否 已 输入 (参见 图 11-8)。 最 后 的 关闭 花 括 号 标志 着 整个 提交 条 件 语 句 的 结尾 。 





Your quotation could not be stored due to a System error. 





Enter your quotation here. 





Add This Quote! 


图 11-7 ”如 果 PHP 脚 本 无 法 找到 quotes . txt 文件， 或 该 文件 不 可 写 ， 用 户 会 看 到 这 个 信息 
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Please enter a quotation! 





Enter your quotation here. 





图 11-8 ”脚本 包括 基本 的 表单 验证 


由 于 该 页 面 处 理 了 表单 之 后 ， 又 再 一 次 显示 了 它 ( 以 便 用 户 能 继续 输入 文本 )， 因 此 并 没有 
将 该 表单 作为 else 语 名 的 一 部 分 显示 出 来 ， 这 和 本 书 中 的 其 他 示例 是 一 样 的 。 

(8) 完成 PHP 广 : 

因为 该 脚本 的 其 余部 分 都 是 标准 的 HTML， 所 以 在 这 里 通过 一 个 PHP 关 闭 标 签 来 退出 PHP 代 
码 段 。 

(9) 创建 HTML 表 单 : 


<form action="add quote.php" method="post"> 
<textarea name="quote" rows="5" cols="30">Enter your quotation here.</textarea><br/> 
<input type="submit" name="submit" value="Add This Quote!" /> 
<input type="hidden" name="submitted" value="true" /> 

</form> 


该 HTML 表 单 显示 了 一 个 文本 框 ， 用 于 输入 文本 。 该 文本 框 具有 预 设 值 Enter your quotation 
here.， 这 是 通过 将 文字 放置 到 textarea 标 签 中 实现 的 。 



































(10) 完成 HTML 页 面 : 
</body> 
</html> 


(11) 将 文件 保存 为 a99_quote .php 并 将 其 放置 在 支持 PHP 的 服务 器 上 的 适当 目录 中 。 

回顾 图 11-2,， 参考 如 何在 服务 器 上 放置 a49_quote .php 和 quotes .txt, 以 及 它们 之 间 的 相 
对 位 置 。 如 果 不 可 能 使 用 这 种 方式 , 或 者 觉得 这 太 混 乱 了 ,也 可 以 将 两 个 文档 放 到 同一 个 目录 中 
(能 够 执行 PHP 脚 本 的 那个 目录 )， 然 后 将 $file 一 行 修改 为 : 

$sfile = 'quotes.txt'; 

(12) 在 Web 训 览 器 中 多 次 运行 该 脚本 (参见 图 11-9 和 图 11-10)。 

(13) 如 果 愿 意 的 话 ， 在 文本 编辑 器 中 打开 auotes . txt 文件， 确认 数据 都 被 写 入 了 。 


Vv 提示 
口 注意 所 有 与 文件 和 目录 相关 的 函数 都 只 能 在 运行 PHP 的 计算 机 ( 即 服 务 器 ) 上 使 用 。 服 务 
器 上 的 PHP 脚 本 无 法 访问 客户 端的 文件 (只 有 将 文件 上 传 到 服务 器 才 可 以 访问 )。 
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口 如 果 在 运行 该 脚本 时 接收 到 了 权限 错误 消息 (参见 图 11-1), 或 者 是 没有 正确 地 设置 权限 ， 
或 者 是 PHP 脚 本 无 法 访问 到 数据 文件 ,这 可 能 是 由 于 文件 名 拼写 错误 或 没有 正确 引用 文件 
在 服务 器 上 的 路 径 。 

口 如 果 正 在 运行 的 PHP 版 本 是 安全 模式 的 ,或 设置 了 open_basedir 指 令 ， 则 在 使 用 PHP 访 
问 文件 和 目录 时 会 受到 一 些 限制 。 请 使 用 phpinfo () 函数 来 查看 服务 器 上 的 这 些 设置 。 
































Your guotation has been stored. 











A book is like a garden carried in the Enter your quotation here. 


pocket. {Chinese Proverb) 























Add This Quote! Add This Quote! 

















图 11-9 填写 表单 图 11-10 ”如 果 一 切 正常 将 会 看 到 该 结果 


























11.3 ”锁定 文件 


尽管 上 一 个 示例 可 以 运行 得 很 好 (希望 如 此 )， 但 它 还 可 以 进一步 改进 。 如 果 同 一 时 刻 只 有 
一 个 单独 的 用 户 提交 表单 ， 则 不 会 有 任何 问题 。 但 如 果 两 个 或 更 多 用 户 在 同一 时 刻 提交 不 同 的 文 
本 呢 ? 当 多 个 PHP 脚 本 试图 在 同一 时 刻写 入 同一 个 文件 时 ， 就 会 出 现 问题 (文件 会 损坏 )。 解 决 
方法 是 在 PHP 对 文件 进行 号 人 时 ， 和 暂时 锁定 该 文件 。 如 果 运 行 的 是 PHP 5.1 及 以 上 版 本 ， 可 以 在 
file_put_contents () 水 数 中 加 入 第 三 个 参数 LOCK_EX 常 量 : 









































file put_ contents ($file, S$data, LOCK_ EX); 

如 果 要 同时 使 用 LOCK_EX 和 FILE_APPEND 常 量 ， 可 以 使 用 二 元 OR 运算 符 (|): 

file put contents ($file, S$data, FILE APPEND | LOCK_EX); 

两 个 变量 的 使 用 顺序 无 关 紧 要 。 如 果 使 用 的 是 早期 的 PHP 版 本 (必须 使 用 fopen () 、fwrite() 
和 fclose () )， 就 需要 在 打开 文件 之 后 调用 flock () 函数 : 


$sfp = fopen($file, 'a+b'); 
flock($sfp, locktype) 


各 种 不 同 的 锁定 类 型 由 表 11-2 中 列 出 的 常量 来 指定 。 当 脚本 在 处 理 文件 时 , 需要 将 文件 解锁 。 


表 11-2 ”flock() 锁 定 类 型 























锁 含义 
LOCK_SH 用 于 读 取 的 共享 锁 
LOCK_EX 用 于 写 入 的 独 享 锁 
LOCK_UN 释放 一 个 锁 


LOCK_NB 非 阻塞 锁 
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例如 ， 要 在 写 入 时 临时 锁定 一 个 文件 ， 可 以 使 用 下 面 的 代码 : 


$fp = fopen($file, 'a+b'); 
flock ($fp, LOCK_ EX); 
fwrite ($fp, S$data); 
flock ($fp, LOCK_UN); 


作为 演示 ， 下 面 我 们 将 重 写 a99_quote.php， 在 写 入 过 程 中 锁定 文件 。 

过 使 用 文件 锁 

(1) 如 果 尚 未 打开 add_quote .php (参看 脚本 11-1)， 在 文本 编辑 器 或 IDE 中 打开 它 。 
(2) 修改 file_put_contents() 行 ,代码 如 下 (参看 脚本 11-2): 


file put_contents($file, $_POST['quote'] . PHP_EOL, FILE_APPEND | LOCK_EX); 

















脚本 11-2 aqq_suote.php 脚 本 的 修改 版 锁定 了 数据 文件 ， 因 此 更 加 安全 可 靠 
































工 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 

6 <title>Add A Quotation</title> 

7 </head> 

8 <body> 

9 <?php // 脚本 11-1 - add_quote.php 

10 /* 脚本 显示 并 处 理 HTML 表 单 。 脚 本 从 文本 框 中 获取 文本 并 将 其 存储 到 文本 文件 中 。 */ 

小 这 

12 // 标识 要 使 用 的 文件 名 : 

13 S$file = '../quotes.txt'; 

14 

15 // 检查 表单 是 否 提 交 : 

16 if ($_SERVER['REQUEST METHOD'] == 'POST') { // 处 理 表单 。 

4 

18 if ( lempty($_POST['quote']) && ($_POST['quote'] != 'Enter your quotation 

here.') ) { // 检查 输入 文本 。 

19 

20 if (is_writable($file)) { // 确认 文件 可 以 写 入 数据 。 

21 

22 file put contents($file, $ POST['quote'] . PHP EOL, FILE APPEND | LOCK EX); 
// 写 入 数据 。 

23 

25 print '<p>Your quotation has been stored.</p>'; 

26 

27 } else { // 无 法 打开 文件 。 

28 print '<p style="color: red;">Your quotation could not be stored due to a 
System error.</p>'; 

29 } 

30 


31 } else { // 没有 输入 文本 。 
32 print '<p style="color: red;">Please enter a quotation!</p>'; 
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33 } 

34 

35 } // 结束 提交 条 件 语句。 

36 

37 // 完成 PHP 节 并 显示 表单 : 

8 

39 

40 <form action="add quote.php" method="post"> 

41 <textarea name="quote" rows="5" cols="30">Enter your quotation here.</textarea><br/> 
42 <input type="submit" name="submit" value="Add This Quote!" /> 
43 </form> 

44 

45 </body> 


46 </html> 


该 命令 给 文件 添加 了 一 个 独 享 镇 ， 因 此 其 他 脚本 不 能 同时 对 其 进行 写 
es 如 果 不 能 使 用 file_put_contents () 和 LOCK_EX， 那 就 需要 应 用 flock () 国 数 : 


flock($fp, LOCK_ EX); 

fwrite($fp, $_POST['gquote'] . PHP_ EOL); 
flock($fp, LOCK_UN); 

fclose(Sftp) ; 


(3) 保存 文件 , 将 其 放置 到 支持 PHP 的 服务 器 上 的 适当 目录 中 ,并 再 次 在 Web 浏 览 器 中 测试 ( 参 
见 图 11-11 和 图 11-12) 。 














Your quotation has been stored . 








Failure is the opportunity to begin again Enter your quotation here. 
more intelligently. {Henry Ford) 























图 11-11 再 次 使 用 该 表单 图 11-12 ”存储 文本 时 仍 不 会 出 现 问 题 
































ww 提示 

口 如 果 使 用 了 flock() 函数 ， 当 调用 fclose() 函数 时 ， 文 件 会 自动 解锁 ， 但 最 好 还 是 在 文 
件 写 入 完成 后 手动 加 上 解锁 语句 。 

口 从 技术 上 讲 ， 如 果 用 追加 模式 打开 了 文件 (如 这 个 示例 所 示 )， 即 便 不 锁定 文件 ， 当 多 个 
脚本 同时 对 文件 进行 写 入 时 也 不 会 产生 什么 问题 。 但 是 ， 这 样 做 安全 性 会 更 高 些 。 

口 要 使 用 锁 ， 则 每 个 脚本 在 写 入 文件 时 都 要 使 用 锁 。 


11.4 ” 读 取 文件 
读 取 文件 的 方法 很 多 ， 应 该 根据 需要 选择 文件 读 取 方式 。 使 用 file_get_contents () 会 将 
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文件 中 的 所 有 内 容 按 照 一 个 字符 串 来 读 取 

$data = file_get_contents($file) ; 

如 果 文 件 每 一 行 都 有 一 些 数据 ， 就 像 例子 中 的 quotes .txt， 最 好 使 用 file() 函数 : 

sdata = file($file); 

file() 函数 是 PHP 中 非常 有 价值 的 一 个 内 置 工具 。 与 file_get_contents () 不 同 , 它 读 取 
文件 中 的 所 有 内 容 并 将 这 些 信 息 放置 在 一 个 数组 中 。 数 组 的 每 个 元 素 都 包含 了 文件 中 的 一 行 ， 这 
些 行 是 用 换行 符 (\n 或 \r\n) 分 隔 的 。 

如 果 $file 指 向 的 文件 包含 两 行 信息 ,每 行 都 是 以 换行 符 结束 的 ， 则 对 应 的 数组 会 包含 两 个 
元 素 。 第 一 个 元 素 等 价 于 $file 中 的 第 一 行 ， 而 第 二 个 元 素 等 价 于 其 中 的 第 二 行 。 如 果 数 据 被 存 
储 到 数组 中 ， 就 可 以 很 方便 地 操作 或 打印 它 了 ， 正 如 在 第 7 章 中 学 到 的 那样 。 

下 面 将 使 用 这 些 知 识 创建 一 个 脚本 ， 随 机 显示 一 条 之 前 存储 的 文本 。 


之 读 取 一 个 文件 























(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 PHP 文 档 , 命 名 为 view_quote .php( 参 看 脚本 11-3): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>View A Quotation</title> 

</head> 

<body> 

<hl>Random Quotation</h1> 





脚本 11-3 view_quote.php 文 件 检 索 文 本 文件 中 所 有 的 文本 ， 并 随机 显示 一 条 














1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>View A Quotation</title> 

了 </head> 

8 <body> 

9 <h1l>Random Quotation</hi1l> 

10 <?php // 脚 本 11-3 - view_quote.php 

11 /* 脚本 显示 并 处 理 HIML 表 单 。 脚 本 从 文本 框 中 获取 文本 并 将 其 存储 到 文本 文件 中 。*/ 
12 

13 // 将 文件 的 内 容 写 入 数组 : 

14 $data = file('../quotes.txt'); 

15 

16 // 获取 数组 中 的 元 素 个 数 : 

17 $n = count ($data); 

18 

19 // 获取 一 个 随机 元 素 : 

20 S$rand = rand(0, ($n - 1)); 
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22 // 打印 文本 : 
23 print '<p>' 。trim(S$data[Srand]) . '</p>'; 


4 
26 </body> 
27 </html> 


(2) 打开 PHP 代 码 段 : 

<?php // 脚本 11-3 - view_quote.php 

(3) 读 取 文 件 内 容 并 将 它们 保存 在 一 个 数组 中 : 

$sdata = file('../quotes.txt'); 

该 函数 将 文件 数据 读 取 到 一 个 名 为 $data 的 数组 中 。s$qdata 的 每 个 元 素 都 是 一 个 字符 串 ， 对 
应 于 一 条 之 前 提交 的 文本 。 

如 果 quotes .txt 文 件 并 不 位 于 该 脚本 的 父 目录 中 ， 请 适当 地 修改 这 里 的 引用 。 

(4) 基于 $data 中 元 素 的 数量 获取 一 个 随机 数 : 


$n = count ($data); 
$srand = rand(0, ($n - 1)); 


第 一 步 获 取 了 $data 数 组 中 元 素 的 数量 (也 就 是 说 ， 有 多 少 条 文本 )。 然 后 使 用 rand () 函数 
选择 一 个 随机 数 ， 需 要 使 用 一 些 逻 辑 。 

如 果 $qata 有 10 个 元 素 , 它们 的 索引 应 该 是 0~9, 所 以 在 这 里 使 用 这 样 一 个 区 间作 为 rand () 
的 参数 。 在 文本 文件 行 数 不 同 的 情况 下 ， 应 该 以 和 $data 中 元 素 的 数量 减 1 作 为 区 间 。 

(5) 打印 文本 : 

print '<p>' . trim($data[l$rand]) . '</p>'; 

一 条 简单 的 print 语 句 连接 了 要 打印 的 随机 文本 。 要 检索 该 文本 ， 需 要 使 用 前 面 生 成 的 
s$rang 数 值 作为 索引 来 引用 $qata 数 组 。 在 这 之 后 截 去 了 该 本 文 末尾 的 换行 符 。 

(6) 完成 PHP 代 码 和 HITML 页面 : 








他 
</body> 
</html> 


(7) 将 文件 保存 为 view_quote .php， 放 置 在 Web 服 务 器 上 (和 adqd_quote.php 放 在 同一 个 
目录 中 )， 并 在 Web 神 览 器 中 测试 (参见 图 11-13)。 











Random Quotation 


Nurture your mind with great thoughts, for you will never go any higher 
than you think. (Benjamin Disraeli) 











图 11-13 每 次 查看 页 面 时 都 会 随机 显示 一 条 文本 
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(8) 在 Web 浏 览 器 中 重新 加 载 页 面 ， 浏 览 另 外 一 条 


随机 观点 (参见 图 11-14)。 








without accepting it. (Aristotle) 


Random Quotation 


Itis the mark of an educated mind to be able to entertain a thought 











w 提示 





PHP 是 否 可 以 读 取 这 个 文件 。 





11.5 ”处 理 文件 上 传 


正如 本 书 已 经 演示 过 的 那样 ， 用 PHP 处 理 HTML 表 单 是 一 个 非常 简单 的 过 程 。 无 论 提 交 的 数 





图 11-14 再 次 访问 view_quote.php 脚 本 会 显示 来 自 文本 文件 中 的 另 一 条 文本 





口 readfile() 函数 用 于 读 取 一 个 文件 并 将 内 容 发 送 到 Web 训 览 器 窗口 。 
口 在 本 章 稍 后 的 部 分 中 ， 还 将 学 到 fgets() 和 fgetcsv() 这 两 种 用 以 读 取 文 件 的 更 为 复杂 


据 是 什么 ，PHP 都 能 简单 并 且 直 观 地 处 理 。 当 用 户 通过 HTML 表 单 上 传 文件 时 也 是 如 此 。 








为 了 让 用 户 能 够 上 传 文件 ， 必 须 对 标准 的 HTML 表 单 作出 3 处 更 改 。 首 先 ， 初 始 form 标 签 必 
须 包 含 代码 enctype="multipart/form-data"， 让 浏览 器 知道 表单 数据 将 具备 不 同 的 类 型 ; 


<form action="script.php" enctype="multipart/form-data" method="post"> 


表单 必须 使 用 POST 方法 。 
其 次 ， 必 须 添加 一 个 特殊 的 隐藏 输入 框 : 


出 





<input type="hidden" name="MAX_ FILE SIZE" Value="30000" /> 


它 用 来 建议 浏览 器 最 大 能 够 上 传 多 大 的 文件 〈 以 字 节 计 )。 





最 后 ， 使 用 file 元 素 创建 所 需 的 表单 字段 (参见 


<input type="file" name="picture" /> 





Browse... 


图 11-15 ”Firefox 这 样 解释 file 类 型 的 输入 框 

















传 到 服务 器 。 上 传 完毕 后 ， 就 能 用 PHP 处 理 该 文件 了 。 





图 11-15 和 图 11-16)。 








Choose File ) no file selected 











图 11-16 Safari Web 浏 览 器 这 样 解释 文件 
输入 框 (选择 文件 的 按钮 在 前 


file 类 型 的 表单 输入 框 允许 用 户 从 他 们 自己 的 计算 机 上 选择 一 个 文件 ， 然 后 提交 ， 从 而 上 














口 如 果 想 要 更 谨慎 一 些 , 可 以 在 调用 file () 函数 之 前 使 用 is_readable () 函数 来 测试 一 下 


面 ) 
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在 PHP 脚 本 中 ， 可 以 引用 $_FILES 变 量 (文件 领域 中 的 $_POST 等 价 物 ) 来 引用 上 传 的 文件 。 
$_FILES 数 组 包含 5 个 元 素 : 
口 name， 文件 在 用 户 计算 机 上 使 用 的 名 字 ， 
口 type， 文 件 的 MIME 类 型 (例如 ，image/ipg); 
D size, 文件 的 大 小 (以 字 节 计 )，; 
口 tmp_name， 文 件 存放 在 服务 器 上 的 临时 文件 名 ; 
D error， 如 果 发 生 了 错误 则 保存 错误 代码 (参见 表 11-3)。 


表 11-3 $_FILES 错 误 代 码 
代 码 意 义 
没有 错误 发 生 
文件 大 小 超过 了 php .ini 中 的 upload_max_filesize 设 置 
文件 超过 了 HTML 表 单 中 的 MAX_FILE_SIZE 设 置 
文件 只 有 部 分 被 上 传 
未 上 传 文件 
临时 目录 不 存在 
写 入 磁盘 出 错 
上 传 文件 的 扩展 名 被 阻止 

注意 ， 错 误 代 码 中 没有 代码 3， 是 不 是 很 奇怪 ? 

当 文 件 上 传 后 ， 服 务 器 会 将 其 放置 在 一 个 临时 目录 中 。 之 后 可 以 使 用 move_uploaded_ 
file() 函数 将 文件 存放 在 其 最 终 目 标 位 置 : 

move _ uploaded file(s$_ FILES['picture']['tmp name'], '/path/to/dest/filename'); 

第 一 个 参数 是 服务 器 中 文件 的 临时 名 。 第 二 个 参数 是 移动 目标 的 完整 路 径 和 文件 名 。 

为 了 使 PHP 能 够 完成 这 些 步 野 ， 必 须 在 php .ini 文 件 中 进行 一 些 配 置 (参见 框 注 “ 配 置 PHP 
以 上 传 文件 ”)， 并 且 Web 服 务 器 必须 对 临时 目录 和 最 终 的 目标 目录 具有 写 权限 。( 默 认 情 况 下 ， 
PHP 对 临时 目录 有 和 写 访 问 权 限 。) 

下 面 编写 一 个 基本 的 脚本 ， 完 成 文件 上 传 并 将 其 存储 到 服务 器 上 。 和 adda_quote.php 脚 本 
类 似 , 该 示例 同时 创建 并 处 理 HTML 表 单 (参见 图 11-17)， 所 有 这 些 都 在 一 个 页 面 中 完成 。 不 过 ， 
首先 需要 在 目标 位 置 创建 一 个 可 写 的 目录 。 














oo ~ 了 了 和 上 mp 一 己 


























Upload a file using this form: 


Choose File ) no file selected 
Upload This File 


图 11-17 该 HTML 表 单 允许 用 户 从 他 们 的 计算 机 上 选择 一 个 文件 上 传 到 服务 器 上 
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(1) 创建 一 个 新 的 文件 夹 ， 名 为 uploads， 放 置 在 Web 根 目录 之 外 (参见 图 11-18) 。 

(2) 使 用 本 章 11.1 布 中 列 出 的 步骤 , 将 该 目录 的 权限 设置 为 所 有 用 户 可 读 、 可 写 和 可 搜索 (在 
Unix 中 是 0777 ) 。 

同样 ， 如 果 运 行 的 是 Windows， 则 可 能 什么 也 不 用 做 (直接 尝试 下 面 的 脚本 看 看 是 否 可 行 )。 
如 果 运 行 的 是 其 他 操作 系统 ， 请 查看 建议 列表 ， 找 到 最 适合 自己 环境 的 方式 。 

















Web 根 目录 的 父 目 录 。 ”Web 根 目录 


一 SEE 
[了 = 一 - ee =: 人 @ upload _file.php 
| : 
| 
| 
| 





uploads 




















图 11-18 ”对 于 这 个 示例 ,必须 具有 一 个 可 写 的 uploads 目 录 。 在 这 里 , 将 其 放置 到 
了 和 Web 根 目录 一 样 的 文件 夹 中 。 因 此 uploads 位 于 upload_file.php 
脚本 所 在 目录 的 上 一 级 目录 ， 并 且 不 能 通过 Internet 访 问 到 


过 使 用 PHP 上 传 文件 


(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 PHP 文 档 ， 命 名 为 uploagd_file.php (参看 脚本 
11-4) ; 





























<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Upload a File</title> 

</head> 

<body> 


脚本 11-4 ”该 脚本 处 理 了 文件 上 传 ， 首先 定义 了 适当 的 HTML 表 单 ， 然 后 调用 了 move_upload- 
eq_file() 将 文件 移动 到 期 望 的 位 置 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.o0org/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Upload a File</title> 

</head> 

<body> 











OOPRODP 
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9 <?php // 脚本 11-4 - upload_ file.php 
/* 脚本 显示 并 处 理 HTML 表 单 。 脚 本 从 文本 框 中 获取 文本 并 将 其 存储 到 文本 文件 中 。*/ 








0 

下 于 

12 if ($_SERVER['REQUEST METHOD'] == 'POST') { // 处 理 表单 。 
3 
4 


1 // 尝试 移动 上 传 文件 : 
15 if (move uploaded file ($ FILES['the file']['tmp name'], "../uploads/ 
{$_FILES['the file']['name']}")) { 





16 

17 print '<p>Your file has been uploaded.</p>'; 

18 

19 }) else { // 移动 出 错 ! 

20 

21 print '<p style="color: red;">Your file could not be uploaded because: '; 
人 人 

23 // 根据 错误 类 型 打印 对 应 的 错误 信息 : 

24 Switch ($_ FILES['the file']['error']) { 

25 case 1: 

26 print 'The file exceeds the upload max filesize setting in php.ini'; 
27 break; 

28 case 2: 

29 Print 'The file exceeds the MAX FILE SIZE setting in the HTML form'; 
30 break; 

31 case 3: 

32 print 'The file was only partially uploaded'; 

33 break; 

34 case 4: 

35 print 'No file was uploaded'; 

36 break; 

37 case 6: 

38 print 'The temporary folder does not exist.'; 

39 break; 

40 default: 

41 print 'Something unforeseen happened.'; 

42 break; 

43 } 

44 

45 print '.</p>'; // 段落 结束 。 

46 

47 } // 结束 move_uploaded_file() 条 件 语 和 句 。 

48 

49 } // 结束 提交 条 件 语 身 。 

50 

51 // 完成 PHP 节 并 显示 表单 : 

S27 “> 

53 

54 <form action="upload file.php" enctype="multipart/form-data" method="post"> 
55 <p>Upload a file using this form:</p> 

56 <input type="hidden" name="MAX FILE SIZE" value="300000" /> 

57 <p><input type="file" name="the file" /></p> 

58 <p><input type="submit" name="submit" value="Upload This File" /></p> 
59 </form> 

60 


61 </body> 
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62 </html> 

(2) 创建 PHP 代 码 段 : 

<?php // 脚本 11-4 - upload_file.php 

(3) 检查 表单 是 否 已 经 被 提交 : 

if (S$_SERVER['REQUEST METHOD'] == 'POST') { 

同样 ， 该 脚本 同时 显示 并 处 理 HIML 表 单 。 如 果 表 单 已 经 被 提交 ， 则 应 该 处 理 上 传 的 文件 。 
(4) 尝试 将 上 传 的 文件 移动 到 最 终 的 目标 位 置 : 


if (move_ uploaded file ($_FILES['the file']['tmp name'], 
>"../uploads/{$_FILES['the file']['name']}")) { 


move_uploaded_file() 函数 尝试 将 上 传 的 文件 (通过 $_FILES['thefile']['tmp_ 
name' ] 指定 ) 移动 到 新 位 置 (../uploads/{$_FILES['thefile']['name']})。 其 位 置 是 
uploads 目 录 , 位 于 当前 脚本 所 在 位 置 的 上 一 级 目录 中 。 文件 的 名 字 将 会 和 用 户 计算 机 上 的 名 字 
相 一 致 。 

将 该 函数 放 在 if 语 句 的 条 件 中 ， 可 以 简单 地 根据 是 否 移动 成 功 做 出 响应 。 

(5) 打印 消息 ， 标 明 操 作 是 否 成 功 : 

print '<p>Your file has been uploaded.</p>'; 

} else { // 移动 出 错 ! 

print '<p style="color: red;">Your file could not be uploaded because: ' 

如 果 移 动 成 功 (参见 图 11-19), 则 执行 第 一 个 print 语 句 。 如 果 没 有 成 功 , 则 应 用 else 子 句 ， 

这 时 会 开始 打印 错误 消息 。 第 6 步 更 明确 地 给 出 了 错误 消息 。 




















Your file has been uploaded. 


Upload a file using this form: 


Choose File ) no file selected 
Upload This File 


图 11-19 ”如 果 上 传 和 移动 文件 成 功 ， 则 打印 一 条 消息 并 再 次 显示 表单 
(6) 如 果 没 有 移动 成 功 ， 则 打印 错误 消息 : 


Switch ($_FILES['the file']['error']) { 
case 1: 
print 'The file exceeds the upload max_ filesize setting in php.ini'; 
break; 
case 2: 
print 'The file exceeds the MAX_ FILE SIZE setting in the HTML form'; 
break; 
case 3: 





























286 0UIHDUO OU0000 





print 'The file was only partially uploaded'; 
break; 

case 4: 
print 'No file was uploaded'; 
break; 

case 6: 
prin 
brea 

defaul 
prin 
brea 


'The temporary folder does not exist.'; 


'Something unforeseen happened.'; 





Pa AN 


} 

如 果 没 有 移动 成 功 , 则 $_FILESI['the_file']['error'] 变 量 中 将 包含 一 个 表示 对 应 的 错 
误 消 息 的 数字 。 通 过 使 用 switch 条 件 语句 ， 访 PHP 脚本 可 以 打印 出 对 应 的 错误 消息 (参见 图 
11-20) 。 








Your file could not be uploaded because: The file exceeds the 
MAX_FILE_SIZE setting in the HTML form. 


Upload a file using this form: 


Choose File ) no file selected 


图 11-20 ”如 果 发 生 错 误 ， 该 脚本 会 指出 错误 原因 
通常 不 应 该 将 这 样 的 代码 放置 到 公开 站 点 中 (其 中 的 信息 太 多 了 )， 但 这 对 于 调试 代码 中 的 
问题 来 说 ， 则 大 有 益处 。 
(7) 完成 错误 消息 ， 关 闭 两 个 条 件 语句 : 


print '.</p>'; // Complete the paragraph. 
} // 结束 move_uploaded_ file() 条 件 语句 。 
】 // 结束 提交 条 件 语句 。 


(8) 退出 PHP 代 码 节 并 创建 HTML 表 单 : 


2 




















<form action="upload file.php" enctype="multipart/form-data" method="post"> 
<p>Upload a file using this form:</p> 
<input type="hidden" name="MAX_ FILE SIZE" value="300000" /> 
<p><input type="file" name="the file" /></p> 
<p><input type="submit" name="submit" value="Upload This File" /></p> 
</form> 


该 HTMI 表 单 非常 简单 ,只 包含 两 个 可 见 的 元 素 : 一 个 fi1e 类 型 的 输入 框 , 和 一 个 提交 按钮。 
与 本 书 中 其 他 HTML 表 单 的 不 同 之 处 在 于 ， 它 使 用 了 enctype 属 性 和 一 个 MAx_FILE_sTZE 隐 藏 
输入 框 。 
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在 为 file 输 入 框 设置 name 属 性 的 时 候 要 很 小 心 ， 因 为 这 个 值 必 须 与 $S_FILES 变 量 中 使 用 的 
索引 精确 匹配 。 在 这 里 使 用 普通 的 the_file。 
(9) 完成 HTML 页 面 : 





















































</body> 
</html> Upload a file using this form: 
(10) 将 页 面 保存 为 upload_file .php， 放 置 在 
支持 PHP 的 服务 器 上 相对 于 uploads 目 录 的 适当 目 ee ey 
录 中 (参见 图 11-18), 并 在 Web 浏 览 器 中 进行 测试 ( 参 (Upload This File ) 
见 图 11-21)。 
由 于 MAX_FILE_SIZE 的 限制 ， 只 有 小 于 300 KB ”图 11-21 在 计算 机 上 选择 一 个 文件 进行 上 传 
的 文件 是 允许 的 。 
(11) 检查 uploads 目 录 ， 确 认 文 件 已 被 放置 到 这 里 。 
Vv 提示 





口 如 果 无 法 移动 文件 并 显示 权限 被 拒绝 错误 ， 请 检查 uploads 目 录 的 权限 。 然 后 检查 脚本 
中 使 用 的 目录 路 径 是 否 正确 ， 并 确定 没有 拼写 错误 。 

口 你 也 许 会 发 现 , 通过 Web 浏 览 器 上 传 的 文件 (从 权限 上 看 ) 是 属于 Web 服 务 器 应 用 程序 的 。 
口 从 安全 的 角度 来 说 ， 最 好 重 命名 上 传 的 文件 。 要 完成 这 一 任务 ， 需 要 设计 一 个 系统 ， 能 














够 生成 一 个 新 的 唯一 的 文件 名 ， 并 在 一 个 文本 文件 或 数据 库 中 同时 保存 原始 文件 名 和 新 
文件 名 。 

口 一 个 脚本 可 以 同时 处 理 多 个 文件 上 传 ， 只 要 它们 的 名 字 不 同 即 可 。 在 这 种 情况 下 ， 需 要 
一 个 MAX_FILE_SIZE 隐 藏 输入 框 来 处 理 。 在 PHP 脚 本 中 ， 需 要 将 move_uploaaded_- 
file() 国 数 应 用 于 $_FILES['filename1']、S$_FILES['filename2']， 依 此 类 推 。 


十 


上 





吕 
国生 
D000 

0 
加 加 
D000 


up 














口 通过 在 PHP 脚 本 中 引用 适当 的 索引 (例如 ，s$_FILES['thefile'] ['size']), 可 以 限 





所 文件 为 确定 的 大 小 或 类 型 (在 文件 上 传 之 后 进行 )。 


D 使 用 unlink () 可 以 删除 文件 而 不 进行 移动 或 复制 。 
口 使 用 copy () 函数 可 以 在 服务 器 上 创建 文件 的 副本 。 





配置 PHP 以 上 传 文件 


Oooooooooooo0ehs.ini000oooooooooooooooooon0 CE 
go0000000000000000pw.inij0000 pinto0)000000 

00 

DOfile uploadas00000000000auploaa tmp airz00000000pPHP 

[ED AL ey ee ee el ee Ce NE La Coal eae 
aL aa wae a 

load max_ filesizel|] post max_ size[| OOO0000000000000UU0 
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MAX FILE _STzz0UUUOUOUOOYwebUOOU0O0O0O0O00O0U0O0000000000 
D000 

加 加 可 和 可可 和 可可 加 和 加 可 可可 本 和 可 加 和 VB 和 加 可 和 本 和 订 加 和 国 让 加 和 全 是 memeeyI mi 加 
max_execution timel]] QODOUUPHPIOOUUOUOUUUOUUUUUO 


11.6 导航 目录 


前 面 的 PHP 脚 本 用 于 处 理 文件 ， 但 使 用 PHP 也 可 以 针对 目录 做 很 多 事情 。 在 这 个 示例 中 ,我 
们 将 编写 一 个 脚本 列 出 目录 的 内 容 ， 但 首先 需要 理解 所 需 使 用 的 各 种 函数 的 用 途 和 语法 。 

要 查找 一 个 目录 中 的 所 有 内 容 ， 最 简单 的 选择 是 使 用 scandir () 国 数 : 

$stuff = scandir($dir); 

该 函数 是 PHP5 中 添加 的 , 用 于 返回 一 个 数组 , 其 中 包含 了 从 给 定 目录 中 找到 的 所 有 项 一 一 目 
录 和 文件 。 就 这 个 与 文件 有 关 的 函数 来 说 ，$dir 的 值 可 以 是 绝对 路 径 也 可 以 是 相对 路 径 。 

如 果 使 用 的 是 旧版 本 的 PHP， 就 需要 使 用 openair () 、zreaddqir() 和 closeqir() 代 替 。 查 
看 PHP 手 册 ， 了 解 xeadaaizr () 函数 的 全 部 语法 和 用 法 。 

下 一 个 例子 会 使 用 scandir() 函数 ， 但 我 们 先 来 看 另外 几 个 函数 。 在 这 个 例子 里 会 用 到 
filesize() 函数 ， 它 用 于 检查 文件 的 大 小 (以 字 布 计算 )。 可 以 将 这 个 值 赋 给 一 个 变量 或 打印 
出 来 : 

$ssize = filesize(s$sfile); 

类 似 地 , filemtime () 国 数 用 于 检索 文件 的 修改 时 间 。 它 会 返回 一 个 时 间 惟 , 可 以 使 用 aate () 
函数 格式 化 这 个 时 间 惟 。 

最 后 ，PHP 包 含 了 很 多 用 于 获取 文件 属性 的 函数 。 本 章 已 经 提 到 过 is_writable() 和 
is_readable(), 此 外 还 有 is_dir() 和 is_file()。 如果 一 个 项 是 文件 或 是 目录 ， 则 对 应 的 函 
i 

下 面 会 将 所 有 这 些 功 能 用 在 同一 个 页 面 中 , 该 页 面 将 形成 一 个 基于 Web 的 控制 面板 ， 用 于 查 
看 目录 的 内 容 (参见 图 11-22)。 


这 创建 目录 控制 面板 




















(GD 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 PHP 文 档 ， 命 名 为 1ist_qdir.php (参看 脚本 11-5): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 











<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 





<title>Directory Contents</title> 
</head> 
<body> 
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Directories 


chl 
ch2 
ch3 
templates 





Files 


Name Size Last Modified 

add_quote.php 1520 bytes December 29,2010 
list_dir.php 1562 bytes December 28, 2010 
login.php 1690 bytes December 28, 2010 
register.php 2339 bytes December 28, 2010 
upload_file.php 1844 bytes December 29,2010 
View_quote.php 773 bytes December 29,2010 

















图 11-22 ”1ist_air.php 脚 本 展示 了 一 个 目录 中 的 内 容 。 上 面 
列 出 了 子 文件 夹 ， 下 面 的 表格 列 出 了 文件 











脚本 11-5 ”该 脚 本 显示 了 一 个 目录 的 内 容 。 首 先 列 出 了 子 文件 夹 ， 然 后 在 一 个 表格 中 列 出 了 文 
件 (包括 文件 大 小 和 修改 时 间 ) 








1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://ww.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>Directory Contents</title> 

7 </head> 

8 <body> 

9 


<?php // 脚本 11-5 - list_dir.php 
/* 脚本 显示 一 个 目录 中 的 文件 和 子 目 录 。*/ 


0 
工 
12 // 设置 时 区 : 
13 date default_ timezone_set('America/New_ York'); 
4 
5 
6 


// 设置 目录 名 并 扫描 其 内 容 : 
$search dir = '.'; 
17 S$contents = scandir($search dir); 








19 // 列 出 目录 中 的 子 目 录 : 
20 // 以 原点 开始 ， 后 接 目 录 名 : 


21 print '<h2>Directories</h2> 


22 <ul>'; 
23 foreach ($contents as $item) { 
24 if ( (is dir($search dir . '/' . $item)) AND (substr($item, 0, 1) != '.') ) { 


25 print "<li>$item</l1i>\n"; 
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26 } 

27 } 

28 

29 print '</ul>'; // 关闭 列表 。 
30 


31 // 创建 一 个 表 头 : 
32 print '<hr /><h2>Files</h2> 


33 <table cellpadding="2" cellspacing="2" align="left"> 


S34 ,ETS 

35 <td>Name</td> 

36 <td>Size</td> 

37 <td>Last Modified</td> 








S38 /tr 

39 

40 // 列 出 目录 下 的 所 有 文件 : 

41 foreach ($contents as $item) { 

42 

43 

44 // 获取 文件 大 小 : 

45 $fs = filesize($search dir . '/' . $item); 
46 

47 // 获取 文件 的 修改 时 间 : 

48 $lm = date('F j, Y', filemtime($search dir . 
49 

50 // 打印 信息 : 

51 print "<tr> 

52 <td>$sitem</td> 

53 <td>s$fs bytes</td> 

54 <td>slm</td> 

55 </tr>\n"; 

56 

57 } // 结束 条 件 语句。 

58 

59  } // 结束 FORERACH。 

60 

61 print '</table>'; // 关闭 HIML 表 格 。 
62 

63 ?> 

64 </body> 

65 </html> 


(2) 打开 PHP 代 码 市 并 设置 时 区 : 


<?php // 脚本 11-5 - list_dir.php 
date_default_timezone_set('America/New_York'); 


因为 该 脚本 要 使 用 aate () 国 数 ， 所 以 需要 建立 一 次 时 区 。 


if ( (is file($search dir . '/' . $item)) AND (substr($item, 0, 1) != '.') ) { 


'/' . $item)); 


见 第 8 章 以 查找 更 多 信息 ,并 


能 从 这 一 章 中 找到 参考 PHP 手 册 中 的 哪 一 部 分 可 以 找到 所 需 


(3) 指定 要 打开 的 目录 ， 并 扫描 其 内 容 : 


$search dir = '.'; 
$contents = scandir($search dir); 


通过 在 PHP 脚 本 的 顶部 将 这 些 值 建 立 为 变量 ， 可 以 根据 需要 很 方便 地 查找 和 修改 这 些 值 。 这 
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里 使 用 圆 点 来 引用 当前 目录 。 也 可 以 使 用 绝对 路 径 (/Users/1larry/Documents 或 C:\\ 
myfiles\\directory) 或 相对 路 径 (. . /myfiles) 来 引用 其 他 目录 ， 只 要 PHP 有 权限 读 这 个 


目录 即 可 。 
第 二 行 扫描 了 目录 的 内 容 ， 并 将 其 作为 一 个 数组 赋 给 变量 Scontents。 
(4) 列 出 该 目录 的 子 目 录 : 
print '<h2>Directories</h2> 
ee 
foreach ($contents as Sitem) { 
If ( (is dir($search dir . '/' . $item)) AND (substr($item, 0, 1) != '.') ) { 


Drint veli>sitems/ Li>\ 
} 
} 


print. Ve/ lL 

foreach 循 环 访问 了 数组 中 的 每 一 项 ， 每 次 都 将 其 中 一 项 赋 给 $item 变 量 。 因 为 首先 希望 列 
出 每 个 子 目 录 , 所 以 使 用 is_qir () 函数 来 确认 项 的 类 型 。 然 后 检查 它 并 不 是 在 当前 目录 (在 Unix 
系统 上 使 用 一 个 单独 的 圆 点 表示 )。 如 果 该 条 件 的 结果 是 IRUE, 则 打印 出 项 的 名 字 , 将 其 放置 在 
列表 项 标签 内 ， 然 后 跟 一 个 换行 符 (使 生成 的 HTML 源 代码 更 整洁 )。 

这 样 在 每 一 项 前 加 上 $search_qdir 和 反 斜 杜 ，is_qdir () 函数 就 也 可 以 用 于 检查 其 他 目录 中 
的 项 了 。 如 果 代码 仅 引 用 $item 变 量 ， 而 没有 添加 目录 路 径 。 那 代码 仅 在 当前 目录 下 生效 。 

(5) 新 建 一 个 标题 ， 并 为 了 罗列 文件 打开 一 个 表格 : 


print '<hr /><h2>Files</h2> 

<table cellpadding="2" cellspacing="2" align="left"> 
> 

<td>Name</td> 

<td>Size</td> 

<td>Last Modified</td> 

/tr 


该 脚本 还 要 显示 文件 的 大 小 和 修改 时 间 。 为 了 使 这 些 信 息 看 起 来 更 漂亮 , 这 里 将 结果 放置 在 
了 HTMIL 表 格 中 。 
(6) 开始 循环 遍历 该 目录 中 的 文件 : 


foreach ($contents as S$item) { 
if ( (is_file($search dir . '/' . $item)) AND (substr($item, 0, 1) != '.') ) { 


这 里 使 用 另外 一 个 foreach 循 环 再 次 遍历 了 目录 的 内 容 。 这 一 次 只 需要 文件 项 (但 不 包括 以 
圆 点 开始 的 隐藏 文件 ) 。 

同样 ， 在 每 个 项 目 之 前 会 添加 $search_dir 和 一 个 反 斜 杠 。 

(7) 计算 文件 的 大 小 和 修改 时 间 ， 并 打印 这 些 信息 : 





























sfs = filesize($search dir . '/' . $item); 

$slm = date('F j, Y', filemtime ($search dir . '/' . $item)); 
DELnE etE> 

<td>$sitem</td> 


<td>s$fs bytes</td> 
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<td>$slm</td> 
/EELSNIY:y 


第 一 行 调用 filesize() 函数 获取 了 文件 的 大 小 (以 字 节 计算 )。 第 二 行 调用 filemtime () 
国 数 ， 它 会 返回 反映 了 文件 修改 时 间 的 时 间 改 。 紧 接着 这 个 值 被 传递 给 了 date () 函数 ， 通 过 进 
行 适 当 的 格式 化 ， 返 回 类 似 November 24, 2011 这 样 的 字符 串 。 最 后 ， 这 两 项 以 及 文件 的 名 字 会 打 
印 在 表格 中 的 适当 列 中 。 

(8) 完成 条 件 语句 和 循环 : 


} 
} 


(9) 关闭 表格 : 
print '</table>'; 
(10) 完成 PHP 代 码 和 HTML 页 面 : 


?> 
</body> 
</html> 


(11) 将 文件 保存 为 1ist_qdir .php， 将 其 放置 在 启用 了 PHP 的 服务 器 上 的 适当 目录 中 ， 并 在 
Web 浏 览 器 中 测试 (参见 图 11-22)。 
(12) 你 可 以 更 改 $search_qir 的 值 ， 并 在 Web 浏 览 器 中 重新 测试 脚本 (参见 图 11-23)。 


























Directories 

e chll 

® Css 

e temp 

e templates 
Files 
Name Size Last Modified 
add_quote.php 1520 bytes December 29,2010 
books.php 514 bytes December 13, 2010 
calculatorhtml © 1094 bytes October 16, 2008 
calculator.php 1709 bytes October 27, 2008 
customize.php 1694 bytes December 20, 2010 
event.html 1043 bytes December 7, 2010 




















图 11-23 ”服务 器 上 另 一 个 文件 夹 的 目录 列表 





w 提示 
口 注意 ， 在 Windows 服 务 器 上 ， 需 要 使 用 双 反 和 斜 线 来 创建 绝对 路 径 名 ， 因 为 Windows 路 径 名 
中 的 单反 斜 线 会 导致 字符 被 转 义 。 因 此 ， 必 须 对 单反 斜 线 进行 转 义 。 
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口 使 用 glob () 函数 可 以 搜索 名 字 与 模式 相 匹配 的 文件 目录 (如 *.jpg 或 filename*.doc ) 。 

口 可 能 会 用 到 的 其 他 文件 函数 包括 fileperms () ， 它 返回 文件 的 权限 ;， fileatime () 用 以 
返回 文件 的 最 后 访问 时 间 ， 以 及 fileowner () ， 它 返回 拥有 该 文件 的 用 户 名 称 。 

口 basename () 或 dirname () 函数 可 以 用 来 返回 完整 路 径 字符 串 中 的 文件 名 或 目录 名 。 

口 finfo_file() 函数 用 于 检测 文件 的 MIME 类 型 。 


11.7 创建 目录 


理解 如 何在 服务 器 上 读 取 和 写 入 文件 只 是 数据 存储 过 程 中 的 一 个 部 分 。 为 了 实现 我 们 的 目 
标 ， 可 能 还 需要 使 用 目录 。 

PHP 中 用 于 创建 目录 的 命令 是 mkdir (): 

mkdir( 'directory name', permissions); 

directory_name 是 要 创建 的 目录 的 名 字 。 这 个 值 可 以 是 以 当前 目录 ( 即 脚 本 所 在 的 目录 ) 
为 起 点 的 相对 路 径 ， 也 可 以 是 完整 路 径 : 

mkdir('C:\\inetpub\\users\\george'); 

在 Windows 服 务 器 上 ， 权 限 会 被 忽略 ， 因 此 该 参数 不 是 必需 的 〈 如 前 面 这 个 示例 )。 在 其 他 
服务 器 上 ， 权 限 的 默认 值 是 0777 (关于 这 个 数值 的 意义 ， 请 参见 本 章 中 11.1 市 )。 

记 住 这 一 点 之 后 , 我 们 来 建立 一 个 脚本 ,为 每 一 个 新 注册 的 用 户 创建 一 个 目录 。 该 脚本 还 会 
将 用 户 的 用 户 名 和 密码 记录 在 一 个 文本 文件 中 , 因此 在 登录 时 可 以 对 用 户 进 行 验 证 。 首先 需要 创 
建 父 目录 (必须 是 可 写 的 ， 这 样 PHP 才 能 在 其 中 创建 子 目 录 ) 和 users .txt 数 据 文件 。 


忆 创建 目录 和 数据 文件 


























(1) 新 建 一 个 名 为 users 的 文件 夹 ， 放 在 Web 根 目录 之 外 。 

可 以 将 其 放置 在 和 之 前 创建 的 uploads 文 件 夹 相同 的 位 置 (参见 图 11-18)。 

(2) 使 用 本 章 第 一 节 给 出 的 步骤 , 将 该 文件 夹 的 权限 设置 为 任何 用 户 可 读 、 可 写 和 可 搜索 (在 
Unix 中 是 0777 ) 。 

如 果 正 在 运行 Windows， 则 很 可 能 不 需要 这 一 步 。 

(3) 在 文本 编辑 器 中 创建 一 个 新 的 空白 文档 。 

(4) 将 该 文件 保存 为 users .Ext 并 放置 在 users 目 录 中 。 

(5) 再 次 使 用 本 章 前 面 列 出 的 步骤 , 将 users .txt 的 权限 设置 为 所 有 用 户 可 写 、 可 读 (在 Unix 



































中 是 0666)。 
如 果 所 用 的 PHP 服 务 器 上 运行 的 是 Windows， 则 这 一 步 也 不 是 必需 的 。 
V 提示 


口 如 果 创 建 了 一 个 PHP 可 以 写 入 的 目录 ,PHP 会 在 该 目录 中 自动 创建 一 个 可 写 的 users .Ext 
文件 。 但 最 好 还 是 不 要 依赖 这 种 自动 的 行为 。 
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局 创建 注册 脚本 








(1) 在 文本 编辑 器 或 IDE 中 打开 一 个 新 的 PHP 文 档 ， 命 名 为 register .php (参看 脚本 11-6): 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 


<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" 











"http://www.w3.org/TR/xhtml1l/DTD/xhtmll-transitional.dtd"> 


<title>Register</title> 

<style type="text/css" media="screen"> 
.error { color: red; } 

</style> 





</head> 
<body> 
<hl>Register</h1> 


脚本 11-6 register .php 脚 本 用 于 两 个 目的 : 将 用 户 信息 记录 到 一 个 文本 文件 中 ， 


\Do、aawm 心 wm 





户 的 内 容 创建 一 个 新 目录 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 


<head> 


<meta http-equiv="content-type" content="text/html; charset=utf-8" 


<title>Register</title> 
<style type="text/css" media="screen'"> 
.error { color: red; } 
</style> 
</head> 
<body> 
<hl>Register</hi1l> 
<?php // 脚本 11-6 - register.php 
/* 脚本 将 用 户 信 息 记录 到 一 个 文本 文件 中 ， 并 为 用 户 创建 一 个 新 目录 。*/ 





// 标识 要 使 用 的 目录 和 文件 : 
$dir = '../users/'; 
$sfile = $dir . 'users.txt'; 


if ($_SERVER['REQUEST METHOD'] == 'POST') { // 处 理 表 单 。 





$Sproblem = FALSE; // 目前 一 切 正常 。 





// 检查 注册 信息 …… 
if (empty($_POST['username'])) { 
Spbroblem = TRUE; 


print '<p class="error">Please enter a username!</p>'; 


if (empty($_POST['password1'])) { 
$sproblem = TRUE; 


print '<p class="error">Please enter a password!</p>'; 


以 及 为 用 


jn 
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34 

35 If ($_POST['password1l'] != $_POST['password2']) { 

36 sproblem = TRUE; 

37 print '<p class="error">Your password did not match your confirmed password!</p>'; 

38 } 

3;9 

40 if (1$problem) { // 如 果 一 切 正常 ……: 

41 

42 if (is writable($file)) { // 打开 文件 。 

43 

44 // 创建 要 写 入 的 数据 : 

45 $subdir = time() . rand(0, 4596); 

46 $data = $_ POST['username'] . "\t" . md5(trim($ POST['password1'])) . 
"\t" . $subdir . PHP_ EOL; 

47 

48 // 写 入 数据 : 

49 file put contents ($file, S$data, FILE APPEND | LOCK_EX); 

50 

51 // 创建 目录 : 

52 mkdir ($dir . $subdir); 

53 

54 // 打印 一 条 消息 : 

55 print '<p>You are now registered!</p>'; 

56 

57 } else { // 无 法 写 入 文件 。 

58 print '<p class="error">You could not be registered due to a System error.</p>'; 

59 } 

60 

61 】 else { // 注册 信息 不 完整 。 

62 print '<p class="error">Please go back and try again!</p>'; 

63 } 

64 

65 } else { // 显示 表单 。 

66 

67 // 结束 PHP 节 并 显示 表单 : 

68 ?> 

69 

70 <form action="register.php" method="post"> 

71 <p>Username: <input type="text" name="username" size="20" /></p> 

72 <p>Password: <input type="password" name="password1" size="20" /></p> 

了 <p>Confirm Password: <input type="password" name="password2" size="20" /></p> 

74 <input type="submit" name="submit" value="Register" /> 

75 </form> 

76 

77 <?php } // 结束 提交 条 件 语 向?> 

78 </body> 


79 </html> 


在 页 面 的 head 中 ,定义 了 一 个 CSS 类 ， 用 于 格式 化 错误 消息 。 
(C) 打开 PHP 代 码 : 


<?php // 脚本 11-6 - register.php 
$dir = '../users/'; 
$file = $dir . 'users.txt'; 
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这 两 个 变量 表示 这 个 示例 正在 使 用 的 目录 和 文件 。 文件 位 于 目录 中 ,所 以 变量 的 值 以 目录 开 
始 。 根 据 你 的 实际 情况 修改 的 $air 的 值 。 
(3) 检查 表单 是 否 已 经 被 提交 : 








if (S$_SERVER['REQUEST METHOD'] == 'POST') { 
该 页 面 再 次 同时 显示 和 处 理 HTML 表 单 。 这 是 通过 一 个 检查 脚本 的 请 求 方式 的 条 件 语句 来 完 
成 的 [e3 
(4) 验证 注册 信息 : 
$sproblem = FALSE; 
if (empty($_POST['username'])) { 
sproblem = TRUE; 
print '<p class="error">Please enter a username!</p>'; 
} 
if (empty($_POST['password1'])) { 
sproblem = TRUE; 
print '<p class="error">Please enter a password!</p>'; 
} 
If ($_POST['password1l'] != $_POST ['password2']) { 
sproblem = TRUE; 


print '<p class="error">Your password did not match your confirmed password!</p>'; 


} 

与 本 书 之 前 开发 的 注册 表单 相 比 ,这 个 注册 表单 是 一 个 更 加 简单 的 版 本 。 这 里 检查 了 提交 的 
用 户 名 和 密码 ， 验 证 的 过 程 和 之 前 开发 的 一 样 。$problem 变 量 用 于 指明 是 否 出 现 了 问题 。 

(5) 检查 错误 : 

if (!$problem) { 

在 这 里 ，$problem 变 量 可 以 用 以 知道 是 否 能 够 注册 该 用 户 。 如 果 没 有 出 现 错误 ， 则 继续 向 
下 进行 将 是 安全 的 。 

(6) 检查 users .txt 是 否 可 以 写 入 数据 : 

if (is writable(s$file)) { 

和 前 面 的 例子 一 样 ， 用 条 件 语句 检查 文件 是 否 可 写 ， 脚 本 可 以 根据 情况 报告 文件 是 否 可 写 。 
如 果 使 用 的 版 本 不 是 PHP 5.1 或 以 上 版 本 ， 在 条 件 语句 中 使 用 fopen () 函数 (参看 11.2 市 )。 

(7) 创建 要 向 文件 中 写 入 的 数据 ， 然 后 将 它 写 入 : 






































$subdir = time() . rand(0, 4596); 
$data = $_POST['username'] . "\t" . md5(trim($_ POST['password1'])) . "\t" . $subdir 
*PHP_EOL; 





file put_ contents ($file, $data, FILE APPEND | LOCK_EX); 

目录 的 名 字 由 一 个 基于 用 户 注册 时 间 的 数字 和 一 个 随机 值 组 成 。 这样 的 系统 有 助 于 保证 目录 
的 名 字 既 是 唯一 的 也 是 有 效 的 。 

这 个 脚本 没有 像 之 前 那样 存储 一 个 单独 的 字符 串 ， 而 是 存储 了 3 块 信息 :用户 名 、 密 码 的 加 
密 版 本 (使 用 mq5 () 函数 ， 请 参见 第 一 个 提示 ) 和 前 面 儿 行 创建 的 目录 名 字 。 首 先 截取 密码 ,去 
掉 了 无 关 的 空格 。 














11.7 0000 297 





为 了 区 分 各 块 信息 ， 在 它们 之 间 插 入 了 一 个 制 表 符 (通过 代码 中 的 \t 生 成 )。 换 行 符 用 于 标 
志 行 的 结尾 仍旧 使 用 PHP_EOL 常 量 。 
(8) 创建 用 户 的 目录 ， 并 打印 一 条 消息 : 


mkdir ($dir . $subdir); 
print '<p>You are now registered!</p>'; 


mkdir () 函数 在 users 目 录 中 创建 了 子 目录 。 目 录 的 名 字 是 前 面 生 成 的 随机 数 。 
(9) 完成 条 件 语句 : 


} else { // 无 法 写 入 文件 。 
print '<p class="error">You could not be registered due to a System error.</p>'; 
} 
) else { // 注册 信息 不 完整 。 
print '<p class="error">Please go back andq try again!</p>'; 











} 
第 一 个 else 完 成 了 判断 是 否 能 够 打开 users .txt 进 行 写 入 的 条 件 语句 (参见 图 11-24)。 第 
二 个 else 完 成 了 判断 用 户 能 否 正确 完成 表单 处 理 的 条 件 语句 (参见 图 11-25)。 











Register 





Please enter a username! 


Register 


Please enter a password! 









































You could not be registered due to a system error. Please go back and try again! 
图 11-24 ”如 果 users .txt 不 可 写 ， 会 出 现 的 结果 图 11-25 ”脚本 报告 表单 所 有 验证 错误 
(10) 向 主 条 件 语句 添加 else 子 句 ， 并 完成 PHP 市 : 
} else { 


该 PHP 脚 本 首先 显示 表单 ， 然 后 处 理 它 ， 这 和 本 章 之 前 的 示例 有 所 不 同 。 之 前 的 脚本 在 处 理 
完 表 单 后 都 会 再 次 显示 该 表单 ， 而 这 个 脚本 不 会 ， 它 会 在 else 语 句 中 再 创建 一 个 表单 。 由 于 页 
面 的 其 他 部 分 只 是 HTML， 因 此 这 里 暂时 离开 PHP 片 段 去 创建 表单 。 

(11) 显示 HTML 表 单 : 


<form action="register.php" method="post"> 
<p>Username: <input type="text" name="username" size="20" /></p> 
<p>Password: <input type="password" name="password1" size="20" /></p> 
<p>Confirm Password: <input type="password" name="password2" size="20" /></p> 
<input type="submit" name="submit" value="Register" /> 

</form> 


(12) 完成 主 条 件 语 句 : 
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<?php } // 结束 提交 条 件 语句 。?> 


最 后 的 右 花 括号 关闭 了 主 提 交 条 件 语句 。 在 使 用 它 之 前 ， 必 须 首 先 在 此 打开 PHP 片 段 。 





(13) 完成 HTML 页 面 : 
</body> 
</html> 


(14) 将 文件 保存 为 register .php, 放置 到 启用 了 PHP 的 服务 器 上 的 适当 目录 中 , 然后 在 Web 
浏览 器 中 测试 (参见 图 11-26 和 图 11-27)。 







































































Register 

Username: |larry 

Password: (++.» 

Cit Pew le = Register 

You are now registered! 
图 11-26 注册 表单 非常 简单 ， 但 功能 齐全 图 11-27 注册 成 功 后 ， 用 户 将 会 看 到 该 消息 
(15) 如 果 需 要 的 话 ， 可 以 在 文本 编辑 器 中 打开 users .txt 文 件 查看 其 内 容 (参看 脚本 11-7)。 





脚本 11-7 Users.txt 列 出 了 3 列 以 tab 划 分 的 字段 信息 用 户 名 、 加 密 的 用 户 密码 以 及 相关 的 


目录 名 
J larry laldc91c907325c69271ddf0c944bc72 12936537501284 
2 john 0cc175b9c0f1b6a831c399e269772661 1293653788455 
3 paul 92eb5ffee6ae2fec3ad71c777531578f 12936537931717 
4 george 4a8a08f09dq37b73795649038408pb5f33 1293653799360 
5 ringo 8277e0910dq750195b448797616e091ad 12936538042144 
w 提示 


口 mdq5 () 函数 创建 了 一 个 散 列 值 ， 它 是 字符 串 经 过 数据 计算 后 的 一 种 表示 。 所 以 这 个 脚本 并 
不 真 的 存储 密码 ,而 是 密码 的 一 个 表示 (从 理论 上 讲 , 不 会 有 两 个 字符 串 具 有 相同 的 ma5 () 
值 )。 很 快 就 会 看 到 这 个 函数 在 登录 脚本 中 的 使 用 方式 。 

D 查看 users 目 录 中 是 否 出 现 了 新 的 子 目录 , 通过 这 种 方式 也 可 以 确认 页 面 是 否 按照 所 期 户 

的 方式 运行 。 

口 如 果 PHP 拥 有 权限 ， 则 可 以 使 用 rmair () 函数 来 删除 现 有 目录 。 


11.8 增 量 读 取 文件 


























在 view_quote.php 脚 本 (参看 脚本 11-3) 中 ， 我 们 使 用 file () 函数 将 整个 文件 读 取 到 了 一 
个 数组 中 。 但 是 如 果 某 一 时 刻 只 需要 读 取 文 件 的 一 小 部 分 呢 ? 这 时 就 需要 使 用 fgets () 函数 了 。 
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fgets () 函数 返回 具有 指定 长 度 的 字符 串 。 通常 将 该 函数 放置 在 while 循 环 中 , 并 用 feof () 
函数 来 确保 没有 到 达 文 件 的 结尾 。 例 如 : 


$fp = fopen($file, 'rb'); 
while (!feof ($fp)) { 
$string = fgets ($fp, 1024); 
} 
fclose ($fp); 


在 这 个 例子 中 ，fgets () 函数 每 次 返回 1023 字 市 的 数据 (指定 的 长 度 1024 再 减 1)， 直 到 行 末 
尾 或 文件 结尾 。 长 度 参 数 是 可 选 的 ,但 如 果 选 用 的 话 ， 它 应 该 使 用 一 个 比 文件 中 每 个 单行 文本 长 
度 都 要 大 的 数值 。 如 果 只 想 读 到 行 的 末尾 ， 省 略 长 度 参数 即 可 : 

$string = fgets ($fp); 

有 的 例子 中 会 采用 更 具 吕 口 口 (delineated) 的 格式 来 存储 数据 (通常 使 用 喜 号 分 隔 ， 因 此 是 
CSV， 这 种 以 辟 号 分 隔 值 的 格式 )， 这 时 可 以 使 用 fgetcsv() 国 数 。 它 使 用 给 定 的 分 隔 符 切 分 字 
符 串 ， 并 返回 一 个 数组 : 


fgetcsv($fp, length, delimiter); 
fgetcsv($fp, 1024); 

















$array 
$array 


前 面 的 函数 调用 也 是 会 返回 1023 字 节 的 数据 , 但 是 它 使 用 默认 的 分 隔 符 (逗号) 对 字符 串 进 
行 切 分 , 该 分 隔 符 用 作 元 素 位 置 的 指示 器 。 该 函数 等 价 于 将 fgets () 和 explode () 函数 结合 到 一 
起 使 用 。 如 果 提 供 分 隔 符 参数 ， 就 可 以 改变 用 于 描述 数据 的 分 隔 符 。 

最 后 ， 因 为 这 些 函 数 依赖 于 行 尾 指 示 符 ， 所 以 最 好 采用 额外 的 保护 措施 ， 即 启用 PHP 的 
auto_detect line endings 设 置 。 可 以 使 用 ini_set() 函数 来 完成 : 





ini_set('auto_ detect_ line endings', 1); 
作为 示例 , 我们 下 面 使 用 前 面 示例 创建 的 users . txt 文件 来 创建 一 个 登录 脚本 。 它 会 持续 地 
读 取 文件 ， 直 到 找到 一 组 匹配 的 用 户 名 和 密码 。 


Bd 








(GD 在 文本 编辑 器 或 IDE 中 打开 一 个 新 的 PHP 文 档 ， 命 名 为 1ogin .php (参看 脚本 11-8): 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Login</title> 

</head> 

<body> 

<hl>Login</hi1l> 








脚本 11-8 login.php 脚 本 使 用 存储 在 users .txt (创建 于 脚本 11-6) 中 的 信息 来 验证 用 户 登 录 


1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
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名 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 

6 <title>Login</title> 

7 </head> 

8 <body> 

9 <h1l>Login</hi1> 

10 <?php // 脚本 11-8 - login.php 

11 /* 脚本 使 用 存储 在 文本 文件 中 的 信息 ， 验 证 用 户 登 录 。*/ 

12 

13 // 标识 要 使 用 的 文件 : 

14 $file = '../users/users.txt'; 

15 

16 if ($_SERVER['REQUEST METHOD'] == 'POST') { // 处 理 表单 。 

17 

18 $loggedin = FALSE; // Not currently logged in. 

19 

20 // 启用 auto_detect_line_settings: 

21 ini set('auto detect line endings', 1); 

2 

23 // 打开 文件 : 

24 $fp = fopen($file, 'rb'); 

2:5 

26 // 循环 遍历 文件 : 

27 while ( $line = fgetcsv($fp, 200, "\t") ) { 

28 

29 // 比较 文件 中 的 数据 和 已 提交 的 数据 : 

30 if ( ($line[0] == $_ POST['username']) AND ($line[1] == md5 (trim($_ POST 
['Password']))) ) { 

31 

32 $loggedin = TRUE // 用 户 名 与 密码 与 文件 中 相 匹 配 。 

33 

34 // 停止 遍历 文件 : 

35 break; 

36 

37 } // 结束 条 件 语 向。 

38 

39 } // 结束 循环 。 

40 

41 fclose($fp); // 关闭 文件 。 

42 

43 // 打印 一 条 消息 : 

44 if ($loggedin) { 

45 print '<p>You are now logged in.</p>'; 

46 } else { 

47 print '<p style="color: red;">The username and password you entered do not 
match those on file.</p>'; 

48 } 

49 





50 } else { // 显示 表单 。 
Bb 
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52 // 结束 PHP 节 并 显示 表单 : 


S37 ?> 

54 

55 <form action="login.php" method="post"> 

56 <p>Username: <input type="text" name="username" size="20" /></p> 
57 <p>Password: <input type="password" name="password" size="20" /></p> 
58 <input type="submit" name="submit" value="Login" /> 

59 </form> 

60 

61 <?php } // 结束 提交 条 件 语句 。?> 

62 

63 </body> 

64 </html> 


(2) 创建 PHP 代 码 段 并 指明 使 用 的 文件 : 


<?php // 脚本 11-8 - login.php 
$file = '../users/users.txt'; 


变量 $file 的 值 应 该 与 register .php 中 的 相同 。 

(3) 检查 表单 是 否 已 经 被 提交 : 

IE ($_SERVER['REQUEST METHOD'] == 'POST') { 

(4) 创建 一 个 临时 变量 ， 用 作 标 志 : 

$loggedin = FALSE; 

$loggedin 变 量 用 于 指出 用 户 是 否 输入 了 正确 的 用 户 名 /密码 组 合 。 当 该 脚本 首次 启动 时 ， 
假设 他 们 还 没有 输入 正确 的 值 。 

(5) 打开 用 于 读 取 的 文件 : 


ini_set('auto detect line endings', 1); 
$fp = fopen(s$file, 'rb'); 


和 file() 函数 不 同 ，fgetcsv () 函数 需要 一 个 文件 指针 。 因 此 ， 必 须 使 用 fopen () 图 数 以 
适当 的 模式 打开 users .txt 文 件 。 在 这 里 ， 使 用 的 模式 是 rbp， 它 表示 文件 需要 以 二 进 制 安全 模 
式 打 开 ， 以 便 读 取 。 

首先 ， 也 是 为 了 安全 起 见 ， 启 用 了 PHP 的 auto_detect_line_encodings 设 置 。 

(6) 循环 遍历 文件 中 的 每 一 行 : 

while ( $line = fgetcsv($fp, 200, "\t") ) { 

该 while 循 环 的 每 一 次 迭代 都 会 读 取 文 件 中 的 200 个 字 节 或 1 行 一 一 看 哪 一 个 先 达 到 。 读 到 的 
数据 将 被 分 割 到 一 个 数组 中 ， 使 用 制 表 符 作为 分 割 元 素 的 分 隔 符 。 

由 于 users .txt 文 件 采 用 的 数据 存储 格式 是 username tab password tab directory newline， 
此 $line 数组 包含 3 个 元 素 ， 索 引 分 别 是 0 (用 户 名 )、1 (密码 ) 和 2 (目录 )。 

(7) 比较 提交 的 值 和 检索 到 的 值 : 


if ( ($line[0] == $_POST['username']) AND ($line[1] == 
>md5 (trim($_POST['password']))) ) { 
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这 个 条 件 语 句 有 两 部 分 , 分 别 检查 了 提交 的 用 户 名 和 存储 的 用 户 名 ($1ine[0]) 是 否 相 等 ， 
以 及 提交 的 密码 是 否 和 存储 的 密码 ($line[1] ) 相等 。 然 而 ， 由 于 存储 的 密码 是 使 用 md5 () 加 
密 过 的 ， 因 此 需要 首先 对 提交 的 值 应 用 ma5 () 然后 再 进行 比较 。 

(8) 如 果 找 到 了 匹配 的 值 ， 将 $l1oggedin 设 置 为 TRUE 并 退出 while 循 环 : 

$loggedin = TRUE; 

break; 

如 果 条 件 值 是 TRUE， 则 提交 的 用 户 名 和 密码 与 文件 中 的 值 是 匹配 的 。 在 这 种 情况 下 ， 
$loggedin 标 志 被 设置 为 TRUE， 而 break 语 句 用 于 退出 while 循 环 。 这 种 系统 的 优点 在 于 ， 只 
需要 读 取 找到 匹配 之 前 的 那 一 块 文件 内 容 。 

(9) 关闭 条 件 语 名 、while 循 环 ， 并 关闭 文件 : 

} 


} 
fclose ($fp); 


(10) 向 用 户 打 印 一 条 消息 : 


if (Sloggedin) { 
print '<p>You are now logged in.</p>'; 
} else { 
print '<p style="color: red;"> The username and password you 
"entered do not match those on file.</p>'; 








} 

该 脚本 使 用 $loggedin 标 志 告 诉 用 户 是 否 “ 已 登入 ”。 在 这 个 过 程 中 还 可 以 添加 一 些 额 外 的 
功能 ， 如 将 用 户 目 录 保 存 到 会 话 当 中 ， 并 将 用 户 带 到 一 个 文件 上 传 页 面 。 

(11) 继续 编写 主 提交 条 件 ， 并 退出 PHP: 


} else { 


Fs 


(12) 创建 HTML 表 单 : 


<form action="login.php" method="post"> 
<p>Username: <input type="text" name="username" size="20" /></p> 
<p>Password: <input type="password" name="password" size="20" /></p> 
<input type="submit" name="submit" value="Login" /> 

</form> 


(13) 返回 PHP 以 完成 主 条 件 语 句 : 


<?php } // 结束 提交 条 件 语句 。?> 


(14) 完成 HTML 页 面 : 
</body> 
</html> 


(15) 将 文件 保存 为 login .php， 放 置 在 启用 了 PHP 的 服务 器 上 的 适当 目录 中 ， 并 在 Web 浏 览 
器 中 测试 (参见 图 11-28、 图 11-29 和 图 11-30)。 
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Login 

Username: |larry 

Password: [ee | Login 

You are now logged in. 

11-28 login 表 单 接收 用 户 名 和 密码 图 11-29 如 果 提 交 的 用 户 名 和 密码 与 之 前 记录 的 

值 匹配 ， 则 用 户 会 看 到 这 样 的 消息 
Login 
The username and password you entered do not match those on file. 
图 11-30 ”如 果 用 户 提交 的 用 户 名 和 密码 与 之 前 记录 的 值 不 匹配 ， 则 会 出 现 这 样 的 结果 
Vv 提示 


口 在 PHP4.2 中 ，fgets () 中 length 参 数 是 可 选 的 并 且 默 认 值 是 IKB (1024 字 节 )， 而 在 PHP 
4.3 中 ，length 参 数 的 默认 值 是 自动 返回 换行 符 之 前 的 所 有 数据 。 

口 在 PHP 5.3 中 ，fgetcsv () 函数 接受 另 一 个 可 选 参数 : 用 于 转 义 问题 字符 的 字符 。 上 默认 的 
转 义 字符 是 反 斜 杠 。 

口 如 果 某 一 行为 空 ，fgetcsv() 返 回 一 个 数组 ， 其 中 只 包含 一 个 单独 的 null 值 。 


.9 回顾 和 实践 
如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
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.9.1 回顾 


口 你 使 用 的 是 哪个 版 本 的 PHP? 

口 要 使 服务 器 上 的 文件 或 文件 夹 可 写 ， 需 要 哪些 步 又? 

口 什么 是 Web 根 目录 (概念 ) ? 你 的 网 站 的 Web 根 目录 是 什么 〈 是 在 你 自己 的 计算 机 上 还 是 
在 远程 服务 器 上 ) ? 

口 向 文件 写 入 数据 的 两 种 方式 各 是 什么 ? 

DO 如 何 向 已 有 文件 中 追加 数据 〈 相 对 于 替换 已 有 数据 而 言 ) ? 

口 如 何 确保 每 个 新 数据 都 独占 一 行 ? 

口 要 让 表单 接受 文件 上 传 ， 在 fozm 开 始 标签 中 应 加 入 什么 属性 ? 
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口 在 哪个 变量 中 PHP 脚 本 能 够 访问 上 传 文件 ?哪个 函数 可 以 将 文件 移动 到 服务 器 的 最 终 目 
标 位 置 ? 
口 fgetcsv() 函数 与 file() 或 file_get_contents() 有 什么 不 同 ? 


11.9.2 ”实践 


查看 PHP 和 手册 中 其 他 与 文件 系统 相关 的 函数 (起 始 网 址 : www.php.net/manual/en/ref. 

filesystem.php ) 。 

修改 add_quote .php， 先 检查 quotes .txt 文 件 是 否 存在 ， 再 检查 文件 是 否 可 写 。 

将 addq_quote.php 页 面 中 的 文本 区 变 为 粘性 文本 区 。 

修改 add_quote.php 用 两 个 文本 框 分 别 接受 引用 语 和 署名 ， 并 写 入 到 文本 文件 中 。 接 下 来 

修改 view_quote .php 使 其 获取 并 显示 这 些 数据 。 

修改 view_quote .php 以 显示 两 个 随机 的 引用 语 。 

修改 uploaq_file.php， 使 其 确认 uploads 文 件 夹 可 写 。 

查看 PHP 手 册 中 glob ( ) 国 数 的 信息 ， 了 解 它 的 功能 和 使 用 方法 。 

更 新 list_dir.php， 显 示 目 录 中 文件 的 其 他 信息 。 

口 在 register .php 中 创建 一 个 验证 系统 ,确保 用 户 名 唯一 。 提 示 : 在 创建 目录 之 前 , 使 用 
PHP 检 查 已 有 用 户 名 列表 ， 查 找 是 否 有 完全 相同 的 用 户 名 。 如 果 没 有 找到 相同 的 用 户 名 ， 
创建 这 个 新 的 用 户 名 。 如 果 这 个 用 户 名 已 经 存在 ,让 PHP 提 示 错 误 信息 , 让 用 户 重 新 创建 
用 户 名 。 

口 结合 使 用 写 入 、 读 取 文 本 文件 以 及 session 或 cookie 技 术 ， 创 建 一 个 真实 的 注册 登录 系统 。 
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本 章 内 容 

口 SQL 介 绍 

口 连接 MySQL 

口 MySQL 错 误 处 理 

口 创建 和 选择 数据 库 
口 创建 表 

口 向 数据 库 插 入 数据 
口 安全 查询 数据 

口 从 数据 库 中 检索 数据 
口 删除 数据 库 中 的 数据 
口 更 新 数据 库 中 的 数据 
口 回顾 和 实践 

















如 果 没 有 数据 库 的 存在 ，Internet 根 本 就 不 会 是 今天 这 个 样子 。 实 际 上 ， 如果 没有 对 多 种 数据 
库 提供 内 置 的 支持 ，PHP 根 本 就 不 会 这 么 流行 或 这 么 有 用 。 本 章 将 使 用 MySQL 作 为 示例 DBMS。 
尽管 MySQL (可 以 应 用 在 多 种 平台 上 ) 也 许 并 不 像 其 他 数据 库 服 务 器 那样 强大 ， 但 对 于 大 多 数 
需求 来 说 ， 它 拥有 足够 的 速度 和 功能 。 此 外 ， 它 对 于 很 多 用 户 来 说 是 免费 的 ， 这 使 它 成 为 Web 开 
发 中 最 常见 的 选择 。 

本 章 介绍 了 如 何 利用 简单 的 数据 库 来 创建 博客 〈 一 种 基于 Web 的 日 记 )。 这 里 学 到 的 东西 足 
够 让 你 达到 入 门 级 别 ， 但 学 习 完 本 章 后 ， 你 若 还 需要 找到 一 些 参 考 资料 来 学 习 更 多 的 内 容 ， 可 以 
先 看 看 附录 B。 


12.1 SQL 介绍 
数据 库 (database) 是 存储 了 信息 的 表 的 集合 〈 表 由 列 和 行 构成 )。 数 据 库 使 用 SQL (结构 化 


查询 语言 ) 来 进行 创建 、 更 新 和 读 取 。 令 人 惊讶 的 是 ，SQL 中 有 些 命令 ( 表 12-1 列 出 了 最 重要 的 
7 个 ) 既是 福 也 是 祸 。 
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表 12-1 常用 SQL 命令 














命 令 功 能 
ALTER 修改 已 有 表 
CREATE 创建 数据 库 或 表 
DELETE 从 表 中 删除 数据 
DROP 删除 数据 库 或 表 
INSERT 向 表 中 添加 记录 
SELECT 从 表 中 检索 数据 
UPDATE 更 新 表 中 的 记录 





SQL 语句 写 起 来 很 像 英语 ， 这 使 得 它 非 常 友好 ,但 要 想 使 用 如 此 有 限 的 术语 来 创建 更 加 复杂 
的 SQL 语句 ， 还 是 要 花 一 些 心思 的 。 本 章 会 讲授 如 何 定 义 所 有 的 基本 SQL 语句 〈 也 称 为 查询 ) 。 

PHP 新 手 可 能 会 对 PHP 和 HTML 的 关系 (PHP 可 以 用 于 生成 HTML, 但 PHP 代 码 从 来 不 在 Web 
浏览 器 中 运行 ) 产生 混 清 。 现 在 又 有 了 数据 库 ， 甚 间 关 系 变 得 更 为 模糊 。 其 过 程 其 实 非常 简单 
PHP 仅 用 来 将 SQL 语句 发 送 到 数据 库 应 用 程序 ,那里 才 是 SQL 语句 真正 执行 的 地 方 。 创 建 表 、 播 
入 记录 、 检 索 到 的 一 些 记录 其 至 是 错误 ， 这 些 执行 的 结果 会 从 数据 库 返 回 到 PHP 脚 本 (参见 图 
12-1)。 














数据 库 应 用 程序 
1. SQL 查 询 
-一 
4 
2. 执行 结果 册 
PHP 脚 本 











图 12-1 PHP 将 SQL 语句 发 送 至 MySQL。MYySQL 将 执行 语句 并 把 结果 返回 给 PHP 脚 本 


考虑 到 这 一 点 , PHP 的 mysql_query () 国 数 将 成 为 本 章 用 到 最 多 的 工具 。 它 可 以 将 一 条 SQL 
命令 发 送 到 MySQL : 

$result = mysqgl_query (SOL command, database connection); 

本 章 开头 就 已 经 说 明了 这 一 点 , 因为 SRL 和 MySQL 的 加 入 使 得 Web 开 发 过 程 更 为 复杂 了 。 当 
发 生 错误 时 (毋庸 置疑 ， 肯定 会 发 生 )， 需要 知道 如 何 最 好 地 对 其 进行 调试 。 

当 PHP 脚 本 没有 与 MySQL 数 据 库 按照 预期 方式 交互 时 , 首先 要 确定 是 不 是 由 查询 本 身 的 问题 
引起 的 (参见 图 12-1 中 的 “SQL 查 询 ”); 还 是 由 查询 的 结果 引起 的 (参见 图 12-1 中 的 “执行 结果 ”)。 
可 以 打印 正在 执行 的 查询 来 调试 脚本 ， 使 用 以 下 代码 : 

print $query; 

$query 代表 一 个 完整 的 SQL 命 令 , 一 般 包含 PHP 变 量 的 值 , 使 用 这 个 仅 有 一 行 的 代码 就 可 以 
显示 正在 运行 的 SQL 语句 。 

接 下 来 ， 使 用 其 他 应 用 程序 执行 上 述 语句 打印 的 SQL 语句 。 这 里 有 两 个 常用 的 工具 : 
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口 MySQL 客 户 端 (参见 图 12-2)， 与 MySQL 交 互 的 命令 行 工具 ， 





人 MMO PHP: Visual QuickStart Guide 


Welcome to the MySQL monitor. Commands end with ; or \g. 
Your MySQL connection id is 4 
Server version: 5.1.44 Source distribution 


Type 'help;' or '\h' for help. Type '\c' to clear the current 
input statement. 


mysqt> 目 











图 12-2 “MySQL 客户 端 是 MySQL 数 据 库 软件 自 带 的 ， 无 需 PHP 脚 本 就 能 执行 查询 
D phpMyAdmin (参见 图 12-3)， 基 于 PHP 的 MySQL 界面。 











phpM uA dmin 思 localhost > 上 myblog 














一 办 Structure , 完 SQL 万 Search 轧 Query 部 Export 可 Import 人 筑 Operations 的 Privileges 陪 Drop 
辐 图 回回] Table ~ Action Records1 Type Collation Size ”Overhead 
Database 口 entries 旧 上 风 加 有 甘 同 Xx 0 MylSAM latin!_swedish ci 1.0 xis 闻 
myblog (]) BS 1 table(s) Sum 0 MylSAM latint_swedish ci 1.0 xris on 
个 Check All/Uncheck All With selected: i] 
目 entries 怠 Print view 电 Data Dictionary 





Name: Number of fields: 


B=—== = 一 


四 | May be approximate. See FAQ 3.11 


| 站 Create new table on database myblog 




















图 12-3 ” phpMyAdmin 可 能 是 用 PHP 编 写 的 最 流行 的 软件 了 。 它 为 MySQL 数 据 库 提供 了 基于 Web 的 界面 


托管 公司 应 该 会 提供 这 两 个 (或 其 中 之 一 ) 工具 , 或 者 你 安装 在 本 地 机 的 软件 中 也 会 提供 这 
些 工 具 。 关 于 使 用 这 两 个 工具 的 说 明 ， 请 参见 附录 A。 
ww 提示 
口 从 技术 上 说 ，DBMS 或 数据 库 应 用 程序 是 与 对 应 的 数据 库 进 行 接口 的 软件 。 不 过 , 大 多 数 
人 认为 数据 库 和 DBMS 是 同义词 。 
口 很 多 其 他 应 用 程序 也 能 脱离 MySQL 客 户 端 和 phpMyAdmin 与 MySQL 进 行 交互 。 有 的 是 免 
费 的 ， 有 的 则 需要 花 钱 。 使 用 Google 快 速 搜 索 MySQL、admin 和 所 用 的 操作 系统 ， 能 够 找 
到 很 多 结果 。 


12.2 ”连接 MySQL 


在 第 11 章 中 操作 文本 文件 时 ， 已 经 了 解 到 一 些 函数 (如 fwrite() 和 fgets()) 在 打开 文件 
时 必须 首先 创建 一 个 文件 指针 (使 用 fopen () )。 之 后 这 个 指针 可 以 用 作 指 向 被 打开 文件 的 引用 。 
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在 使 用 数据 库 时 ， 采 用 类 似 的 过 程 。 首 先 ， 需 要 建立 一 个 到 数据 库 服务 器 (这 里 是 MySQL) 的 
连接 ， 之 后 该 连接 将 可 以 用 作 其 他 命令 的 访问 点 。 连 接 到 数据 库 的 语法 是 : 

$dbc = mysql_ connect (hostname, username, password); 

数据 库 连 接 ($dbc) 的 建立 至 少 需要 3 个 参数 : 主机 名 (一 般 是 localhost)、 用 户 名 和 与 用 户 
名 对 应 的 密码 。 

如 果 使 用 的 是 主机 公司 的 数据 库 , 该 公司 很 有 可 能 会 提供 用 户 名 和 密码 。 如 果 是 在 自己 的 计 
算 机 上 运行 MySQL， 请 参见 附录 A 学 习 如 何 创建 用 户 。 

一 旦 使 用 完 数据 库 ， 就 可 以 关闭 数据 库 连接 ， 就 像 关闭 一 个 已 打开 的 文件 一 样 : 

mysql_close(s$sdbc); 

PHP 脚 本 会 在 脚本 执行 结束 后 自动 关闭 数据 库 连 接 , 但 最 好 是 在 不 使 用 时 显 式 地 关闭 数据 库 

ee 简单 的 脚本 ， 堂 试 连接 到 MySQL。 当 这 个 连接 可 
以 工作 之 后 ， 就 能 行 本 章 的 其 余部 分 


> 0 





























tt 














(1) 在 文本 编辑 器 或 IDE 中 启动 一 个 新 的 PHP 文 档 ， 命 名 为 nysql_connect .php (参看 脚本 
12-1) 。 

<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Connect to MySQL</title> 

</head> 

<body> 














脚本 12-1 能 够 连接 到 MySQL 服 务 器 是 最 重要 的 一 步 ， 这 个 脚本 测试 了 这 个 过 程 


1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>Connect to MySQL</title> 

7 </head> 

8 <body> 

<?php // 脚本 12-1 - mysql_connect.php 

10 /* 脚本 连接 到 MySQL 服 务 器 。*/ 


Ko) 


12 // 尝试 连接 到 MySQL 并 打印 消息 : 
13 if ($dbc = mysql connect('localhost', 'username', 'password')) { 


15 print '<p>Successfully connected to MySQL!</p>'; 


17 mysql_close ($dbc); // 关闭 连接 。 
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18 

19 } else { 

20 

必 屯 print '<p style="color: red;">Could not connect to MySQL.</p>'; 
22 

237 月 

24 

25 ?> 

26 </body> 

27 </html> 

(2) 打开 PHP 代 码 片 段 : 


<?php // 脚本 12-1 - mysql_connect.php 
(3) 连接 到 MySQL， 并 报告 结果 : 


if ($dbc = mysql_ connect('localhost', 'username', 'password')) { 
print '<p>Successfully connected to MySQL!</p>'; 
mysql_close(s$dbc); 

} else { 
print '<p style="color: red;">Could not connect to MySQL.</p>'; 

} 


通过 将 尝试 连接 的 代码 放 到 if-else 语 句 的 条 件 中 ， 可 以 报告 连接 是 否 能 够 工作 。 

本 章 将 继续 使 用 username 和 password 作 为 用 户 名 和 密码 。 对 于 所 编写 的 脚本 ， 需 要 使 用 Web 
主机 提供 的 值 来 替换 它们 ， 或 使 用 附录 A 中 列 出 的 步骤 添加 一 个 用 户 ， 并 将 用 户 名 和 密码 设置 到 
这 里 。 

如 果 连 接 成 功 建立 ， 则 会 打印 一 个 正确 消息 并 关闭 连接 。 否 则 , 会 打印 一 个 错误 消息 ,而 无 
需 关闭 数据 库 连 接 (因为 根本 就 没 打开 )。 

(4) 完成 PHP 代 码 和 HTML 页 面 : 


?> 
</body> 
</html> 


(5) 将 文件 保存 为 mysql_connect .pnp，, 放置 在 启用 了 PHP 的 计算 机 中 适当 的 目录 中 ,然后 
在 Web 浏 览 器 中 测试 它 (参见 图 12-4)。 


Si © (Qy Connect to MySQL x \ a 


€ > |® phpvqs4:8888/mysql_connect.php 



































Successfully connected to MySQL.! 























图 12-4 ”如 果 PHP 支 持 MySQL， 并 且 使 用 的 用 户 名 /密码 /主机 这 个 组 合 三 
是 正确 的 ， 就 会 看 到 这 个 简单 的 消息 
如 果 看 到 了 类 似 图 12-5 中 的 结果 ， 请 再 次 检查 用 户 名 和 密码 值 。 这 些 值 必须 和 所 用 的 Web 主 
机 提供 的 值 或 在 创建 用 户 时 使 用 的 值 一 致 。 也 可 以 在 MySQL 客 户 端 (参见 附录 A) 中 测试 连接 用 
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的 用 户 名 和 密码 。 
OO 











(yy Connect to MySQL x YA 四 


fe > 已 1®@ phpvqs4:8888/mysql_connect.php 





Warning: mysql_connectO [function.mysqgl-connect]: Access denied for user 
‘usernasme'@'localhost' (using password: YES) in 


/Users/larryullman/Sites/phpvqs4/mysql_connect.php on line 13 








Could not connect to MySQL. 








图 12-5 ”如 果 PHP 无 法 连接 到 MySQL， 可 能 会 看 到 类 似 这 样 的 消息 。 错 误 
消息 可 能 出 现 也 可 能 不 出 现 ， 它 取决 于 当前 的 错误 管理 设置 











如 果 看 到 call to undefined function mysql_connect... 这 样 的 消息 ， 可 能 是 由 于 所 用 的 PHP 版 本 
不 支持 MYSQL (参见 框 注 “PHP 中 对 MySQL 的 支持 ”)。 


PHP 中 对 MySQL 的 支持 
DUDDPHPYMySQLYI OUUUDUPHPIUUUU MySQLUOOOOOOO0OO0OO0ODO0D 
加 
MySQLUUOUOOOUOOOO0OUPHPUUOUUDODO 
DUO000000000000000...ndefinedfoncionmysql OOUOOO0OO0OO0000 
UUUPHEUOO0OO0OD0UMYSQLOUOOUOUUUOUOUUUUOUOUOO0O0O00000D0 艺 


U0MysQIUUOUOUOUOUUOOUOOOOOOUOUOUUOUOUOOU0O0O0O0 
DUDPHPUUD 


Vv 提示 

口 很 多 mysql_something() 函数 中 的 数据 库 连 接 参 数 (本 例 中 是 $dbc) 都 是 可 选 的 。 但 
本 书 中 的 所 有 示例 都 没有 省 略 它 ， 就 像 它 在 mysqli_something() 函数 中 不 是 可 选 参 数 
一 样 。 接 下 来 你 也 会 用 到 mysqli_something() 函数 ， 到 时 需要 提供 数据 库 连 接 (参见 
框 注 “MySQL 扩 展 ”)。 











MySQL 扩 展 


PHPUOUOUODOOUODOOOUOODOODODO MysQUOOUOOOOOO0ODO0O0MysSQUOOUODODO 
D00000000000000000000PHPYU MySQLIOUUOUU MysQLUODD 
DODmysal (0000 


UU00000 MysQ5a0000 MysQLUUDOOO0OO0OO0ODURPHEPEsSUUOOOUUUO0DO 
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MySQL41I000000000000myssiUUUU00U0U0MysQUUOUO0OU000D0 
DDO000000000MySQOGOOOOUO0O000000000000000U00000 
U000 MysQiUUOOOUOO0O0OU0O0ORPHPEUUOUOUOOUOPHPo0 MySQL500000 
UO 


口 只 有 当 PHP 脚 本 和 MySQL 数 据 库 位 于 同一 台 计 算 机 上 时 ， 才 能 使 用 localhost 作 为 主机 名 。 
通过 修改 PHP 脚 本 中 的 主机 名 ， 并 在 MySQL 中 建立 适当 的 权限 ， 可 以 使 用 PHP 连 接 到 运 
行 于 远程 服务 器 上 的 MySQL 数 据 库 。 

口 PHP 对 很 多 数据 库 (包括 dBase、FilePro、mSQL、MySQL、Oracle、PostgreSQL 和 Sybase) 
都 有 内 置 的 支持 。 如 果 使 用 的 数据 库 类 型 不 受 直接 支持 (例如 ，Access 或 SQL Server)， 
那么 可 以 使 用 PHP 的 ODBC (开放 数据 库 连 接 ) 函数 以 及 数据 库 的 ODBC 了 驱动 来 与 数据 库 
交互 。 

口 现在 组 合 使 用 PHP 和 MySQL 非 常常 见 ， 所 使 用 的 服务 器 可 能 同时 配置 了 PHP 和 MySQL: 
LAMP 、MAMP 和 WAMP。 它们 分 别 对 应 着 不 同 的 操作 系统 (Linux、Mac OS X 和 Windows ) 
上 的 Apache Web 服 务 器 、MySQL DBMS 以 及 PHP。 

口 本 章 使 用 的 是 MySQL， 所 以 用 到 的 所 有 函数 都 是 MySQL 相 关 的 。 例 如 ， 要 连接 到 MySQL 
中 的 数据 库 ， 可 能 使 用 mysql_connect () 函数 ， 但 如 果 使 用 的 是 PostgreSQL， 则 需要 使 
用 pg_connect () 来 代替 它 。 如 果 没 有 使 用 MySQL DBMS ， 请 使 用 PHP 和 手册 (可 以 在 
www.PHPnet 找 到 ) 查找 对 应 的 函数 名 。 


12.3 ”MySQL 错误 处 理 


本 章 在 深入 介绍 如 何 使 用 MySQL 之 前 ， 最 好 提前 讨论 一 些 用 于 处 理 错 误 的 技术 。 会 遇 到 的 
常见 错误 有 : 
口 连接 MySQL 失 败 ， 
口 选择 数据 库 失败 ， 
口 无 法 运行 查询 ; 
D 查询 没有 返回 结果 ， 
口 数据 没有 插入 到 表 中 。 

根据 经 验 ， 可 以 知道 为 什么 通常 会 发 生 这 些 错误 , 但 立即 了 解 一 下 在 运行 脚本 时 发 生 了 什么 
错误 ， 便 可 以 节省 不 少 调试 时 间 。 要 想 通 过 脚本 显示 关于 所 发 生 错误 的 详实 报告 ， 请 使 用 
mysql_error() 国 数 。 该 函数 可 以 返回 MYSQL 服务 器 返回 关于 错误 的 文本 信息 。 

有 了 该 函数 ， 可 能 还 需要 使 用 一 些 PHP 工 具 来 处 理 错 误 。 尤 其 是 错误 控制 运算 符 (@)， 当 在 
一 个 函数 名 之 前 使 用 该 运算 符 时 ， 可 以 阻止 函数 调用 可 能 产生 的 任何 错误 消息 或 警告 : 

@function name(); 


注意 ， 该 运算 符 并 不 能 阻止 错误 的 发 生 ， 它 只 是 防止 立即 显示 出 错误 消息 。 只 有 在 发 生 错误 
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时 希望 自己 处 理 时 ， 才 使 用 该 运算 符 。 


信 使 用 错误 处 理 








(1) 在 文本 编辑 器 或 IDE 中 打开 mysql_connect .php (参看 脚本 12-1)。 
(2) 像 下 面 这 样 (参看 脚本 12-2) 修改 if 条 件 , 阻止 由 mysql_connect () 函数 产生 的 任何 PHP 


错误 : 


站 








(Sdbc = @mysql connect('localhost', 'username', 'password')) { 


脚本 12-2 通过 向 脚本 中 添加 错误 控制 (@ 符 号 和 mysql_error() 函数 ), 可 以 更 有 目的 地 处 理 


‘OOORODOP 


i 
OARWO PO 





MD N IN 
N 是 口 


N ND ND N N 
OOOO 


发 生 的 错误 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Connect to MySQL</title> 

</head> 

<body> 

<?php // 脚本 12-2 - mysql_connect.php #2 

/* 脚本 连接 到 MySQL 服 务 器 。*/ 


// 尝试 连接 到 MySQL 并 打印 消息 : 
if ($dbc = @mysql connect('localhost', ‘username', 'password')) { 


print '<p>Successfully connected to MySQL!</p>'; 

mysql_close ($dbc); // 关闭 连接 。 
} else { 

print '<p style="color: red;">Could not connect to MySQL:<br/>'.mysql error().'</p>'; 
} 


?> 
</body> 
</html> 


这 里 使 用 e 符 号 阻止 了 错误 消息 , 当 mysql_connect () 函数 发 生意 外 时 (参见 图 12-5), PHP 
不 会 打印 出 错误 消息 。 错 误 仍然 会 发 生 ， 下 一 步 的 更 改 会 处 理 它们 。 
(3) 在 else 部 分 中 的 print 语 句 中 添加 mysql_error () 国 数 : 





print '<p style="color: red;">Could not connect to MySQL:<br/>' . mysql error() . '.</p>'; 
该 脚本 不 是 打印 错误 消息 或 依赖 PHP 处 理 错误 (参见 图 12-5)， 而 是 在 当前 上 下 文中 打印 





MySQL 错 误 。 这 是 通过 将 一 些 HTML 和 mysql_error () 函数 连接 在 一 起 做 到 的 。 
注意 mysql_error () 函数 ， 在 这 个 例子 中 ， 没 有 提供 数据 库 接 连 参数 $dbpc， 因 为 没有 创建 


数据 库 连 


接 。 
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(4) 保存 文件 并 在 Web 浏 览 器 中 再 次 测试 (参见 图 12-6)。 


800 (Connect to MySQL be AR 


AUCG phpvgs4:8888/mysql_connect.php ' 








Could not connect to MySQL.: 
Access denied for user usernasme @ localhost (using password: YES). 














12-6 使 用 PHP 的 错误 控制 函数 ， 可 以 调整 对 错误 进行 处 理 的 方式 


如 果 发 生 了 错误 ， 其 结果 现在 看 起 来 要 比 图 12-5 好 一 些 。 如 果 脚 本 连接 正常 ， 结 果 看 上 去 会 
是 如 图 12-4 所 示 的 那样 ， 因 为 此 时 并 没有 调用 错误 管理 工具 。 

w 提示 
口 在 本 章 中 , 显示 错误 消息 只 是 为 了 协助 调试 过 程 。 真正 的 Web 站 点 不 应 该 将 错误 消息 明确 
地 显示 给 用 户 。 
口 可 以 使 用 e 符 号 阻止 来 自任 何 国 数 的 错误 、 通 知 或 警告 ， 而 不 只 是 MySQL 相 关 的 函数 。 例 
如 : 


@include('./filename.php'); 

口 你 可 以 看 到 ， 当 发 生 连 接 错误 时 ， 还 可 以 调用 die () ， 这 是 exit () 的 同义词 。 其 背后 隐 
含 的 思想 是 ， 如 果 数 据 库 连 接 无 法 建立 ， 则 程序 不 应 继续 运行 。 本 章 中 省 略 了 这 些 内 容 ， 
因为 这 既 策 重 又 容易 使 用 不 当 。 


12.4 创建 和 选择 数据 库 


在 PHP 脚 本 能 够 与 一 个 数据 库 进 行 交 互 之 前 ， 必 须 首 先 选 中 所 需 的 数据 库 。 当 然 ， 为 了 选中 
一 个 数据 库 ， 该 数据 库 必须 存在 。 可 以 使 用 PHP、MySQL 客 户 端 、 phpMyAdmin 或 其 他 很 多 工具 
来 创建 数据 库 ， 只 要 使 用 的 MySQL 主 机 名 /用 户 名 /密码 的 组 合 具 备 这 样 做 的 权限 。 

数据 库 权 限 比 文件 权限 要 稍微 复杂 一 些 , 但 需要 理解 这 样 一 个 事实 一 一 不 同类 型 的 用 户 可 以 
被 赋予 不 同 的 数据 库 权 限 。 例 如 ， 一 个 DBMS 用 户 也 许可 以 创建 新 数据 库 或 删除 现 有 数据 库 (所 
用 的 PDBMS 中 可 能 有 大 量 的 数据 库 ) ， 但 一 个 低级 用 户 也 许 只 能 在 一 个 单独 的 数据 库 中 创建 或 修 
改 表 ， 而 更 多 的 基本 用 户 也 许 只 能 从 表 中 读 取 数据 ， 但 不 能 修改 表 。 

如 果 正 在 托管 站 点 中 使 用 PHP 和 MySQL， 则 主机 托管 公司 通常 会 给 予 你 第 二 种 类 型 的 访 
问 一 一 控制 一 个 单独 的 数据 库 而 不 是 DBMS 本 身 一 一 并 建立 好 初始 数据 库 。 如 果 正 在 使 用 自己 的 
服务 器 ， 或 者 具备 管理 员 权限 ， 则 可 以 创建 新 用 户 和 数据 库 。 

要 使 用 PHP 创 建 数据 库 ， 可 以 使 用 mysql_query() 函数 结合 CREATE DATABASE 


databasename SQL 命令 : 



























































mysql_query('CREATE DATABASE somedb', S$dbc); 
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当 创 建 完 数据 库 后 ， 可 以 使 用 mysql_select_qb() 函数 来 选中 该 数据 库 : 

mysql_select db('somedb', $dbc); 

注意 , 任何 时 候 一 个 数据 库 只 需要 创建 一 次 , 但 每 次 在 其 上 运行 查询 之 前 都 需要 选中 它 。 换 
句 话 说， 一些 开发 人 员 需 要 执行 前 述 的 第 一 步 〈 指 创建 数据 库 ) ， 而 每 个 人 在 每 个 PHP 脚 本 中 都 
需要 执行 前 述 第 二 步 〈 指 选中 数据 库 ) 。 

在 这 个 例子 中 , 将 创建 一 个 新 的 数据 库 并 选中 它 。 再 次 重申 ,创建 数据 库 的 查询 需要 具备 管 
理 员 访问 权限 。 如 果 Web 主 机 对 访问 进行 阻止 ， 则 它 应 该 为 后 续 请 求 创建 初始 数据 库 ， 这 时 只 需 
编写 该 脚本 的 第 二 部 分 一 一 选择 数据 库 。 

字 创建 并 选中 数据 库 






































(1) 在 文本 编辑 器 或 IDE 中 打开 mysql_connect .php (参看 脚本 12-2)。 
(2) 如 果 需 要 的 话 ， 在 第 一 个 print 语 句 后 创建 新 的 数据 库 (参看 脚本 12-3): 


if (@mysql_query('CREATE DRATABASE myblog', $dbc)) { 
print '<p>The database has been created!</p>'; 




















} else { 
print '<p style="color: red;">Could not create the database because:<br/>' 
"mysql_error($dbc) . '.</p>'); 














脚本 12-3 通过 3 步 创建 了 一 个 新 的 数据 库 ， 连接 到 数据 库 ， 使 用 mysql_query () 函数 运行 
CREATE DATABASE 查 询 ， 然 后 关闭 连接 





hs <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.o0rg/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
3 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 
5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>Create the Database</title> 
7 </head> 
8 <body> 
9 <?php // 脚本 12-3 - create_db.php 
0 ”/* 脚本 连接 到 MySQL 服 务 器 。 创 建 并 选中 数据 库 。*/ 
1 
12 // 尝试 连接 到 MySQL 并 打印 消息 : 
13 if ($dbc = @mysqgl connect('localhost', 'username', 'password')) { 
4 
5 print '<p>Successfully connected to MySQOL!</p>'; 
16 
a // 尝试 创建 数据 库 : 
18 if (@mysql query('CREATE DATABASE myblog', $dbc)) { 
19 print '<p>The database has been created!</p>'; 
20 } else { // 无 法 创建 数据 库 。 
21 print '<p style="color: red;">Could not create the database because:<br/>' . 
mysql_error($dbc) . '.</p>'; 
22 } 
23 


24 // 尝试 选中 数据 库 : 
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25 if (@mysql select db('myblog', $dbc)) { 


26 print '<p>The database has been selected!</p>'; 

27 } else { 

28 print '<p style="color: red;">Could not select the database because:<br/>' . 
mysql error($dbc) . '.</p>'; 

29 } 

30 

31 mysql_close ($dbc); // 关闭 连接 。 

32 

33 } else { 

34 

35 print '<p style="color: red;">Could not connect to MySQL:<br/>' . mysql error() . '.</p>'; 

36 

37 } 

38 

39:。?> 

40 </body> 


41 </html> 


如 果 需 要 创建 数据 库 ， 使 用 该 构造 来 清晰 有 效 地 完成 这 一 任务 。 使 用 mysql_query () 函数 
来 运行 CREATE DATABASE myblog 这 个 查询 。@ 符 号 用 于 阻止 错 误 消息 ， 转 而 在 else 子 句 中 通 
过 连接 print 和 mysql_error () 函数 来 处 理 该 错误 。 

注意 ， 这 个 mysql_error () 调 用 可 以 提供 特定 的 数据 库 连 接 参 数 ，$gbc。 

如 果 已 经 创建 了 该 数据 库 ， 请 跳 过 这 一 步 。 

(3) 尝试 选中 该 数据 库 : 


if (@mysql_select db('myblog', S$dbc)) { 
print '<p>The database has been selected!</p>'; 











} else { 
print '<p style="color: red;"> Could not select the database because:<br/>' 
"mYySql_error($dbc) . '.</p>'; 














这 个 条 件 语 句 和 第 2 步 中 的 结构 一 样 。 如 果 PHP 能 够 选中 数据 库 ， 则 会 打印 一 条 消息 。 如 果 
不 能 选中 该 数据 库 ， 则 会 转 而 显示 特定 的 MySQL 错 误 。 

PHP 在 数据 库 上 运行 查询 的 任何 脚本 都 必须 在 连接 到 MySQL 并 选中 数据 库 后 才能 工作 。 

(4) 如 果 需 要 的 话 ， 可 以 修改 页 面 的 标题 来 反映 脚本 的 新 目的 : 


<title>Create the Database</title> 


(5) 将 脚本 保存 为 create_db.php, 放置 在 启用 了 PHP 的 服务 器 中 适当 的 目录 中 , 然后 在 Web 
浏览 器 中 测试 (参见 图 12-7 和 图 12-8)。 











Successfully connected to MySQL.! 


The database has been created! 





The database has been selected! 














图 12-7 如果 数据 库 可 以 创建 并 选中 ， 将 会 在 浏览 器 中 看 到 这 样 的 结果 
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Successfully connected to MySQL'! 


Could not create the database because: 
Access denied for user username'@ localhost to database Imyblog . 


Could not select the database because: 
Unknown database "myblog . 











图 12-8 ”如 果 用 户 没 有 授权 创建 数据 库 ， 将 会 看 到 这 样 的 消息 。 如 果 没 有 
权限 选中 数据 库 ， 也 会 出 现 类 似 的 结果 

















w 提示 

口 可 能 不 需要 频繁 地 创建 数据 库 , 而 且 道 常 也 不 会 使 用 PHP 脚 本 来 创建 数据 库 。 不 过 , 该 实 

例 同时 演示 了 如 何 使 用 PHP 执 行 简单 的 查询 和 创建 数据 库 所 需 的 SQL 命 令 。 

口 通常 ， 在 设置 数据 库 信 息 (主机 和 名、 用户 名 、 密 码 和 数据 库 名 称 ) 时 ， 最 好 使 用 变量 或 
常量 ,但 这 些 例子 里 并 没有 这 么 做 。 使 用 变量 或 常量 ， 可 以 将 这 些 代码 插入 到 适当 的 函 
数 中 。 这 样 做 还 可 以 将 数据 库 特 定 的 代码 同 脚本 的 功能 性 代码 分 离开 ， 易 于 将 这 些 代码 
移植 到 其 他 应 用 程序 中 。 


12.5 创建 表 


在 创建 并 选中 了 初始 数据 库 之 后 ,就 可 以 开始 在 其 中 创建 单独 的 表 了 。 一 个 数据 库 可 以 由 多 
个 表 构 成 ， 但 在 这 个 实例 中 只 创建 一 个 用 于 存放 数据 的 表 。 

要 在 数据 库 中 创建 表 , 需要 使 用 SQL 这 种 数据 库 能 够 理解 的 语言 。 因 为 SQL 非常 像 英 语 口 语 ， 
所 以 创建 新 表 的 查询 如 下 所 示 : 

CREATE TABLE tablename (columnl1 definition, column2 definition, etc.) 

每 个 列 之 间 用 逗号 分 隔 , 首先 指定 列 的 名 字 , 然后 是 列 类 型 。 常 见 的 类 型 有 TEXT、VARCHAR 
( 变 长 字符 串 )、DATETIME 和 INT (整数 )。 

强烈 建议 将 第 一 个 列 创建 为 主键 (primary key) (用 于 引用 整个 行 的 列 )， 因 此 一 个 简单 的 


CREATE 语 名 是: 


















































CREATE TABLE my _table ( 
id INT PRIMARY KEY, 
information TEXT 


) 

表 的 主键 是 一 个 特殊 列 , 由 用 于 引用 表 中 整个 行 的 唯一 值 构 成 。 数据库 会 为 该 列 创建 一 个 索 
引 ， 以便 更 快 地 在 表 中 导航 。 一 个 表 只 能 有 一 个 主键 ,通常 将 其 设置 为 自动 增长 的 整数 列 。 其 第 
一 行 的 键 值 是 1， 第 二 行 的 键 值 是 2， 依 此 类 推 。 引 用 键 值 永远 都 可 以 检索 到 对 应 行 的 数据 。 

可 以 访问 MySQL 网 站 查看 关于 SQL 的 更 多 信息 。 然而, 按照 这 一 市 的 介绍 , 将 能 够 完成 基本 
的 数据 库 任务 。 表 12-2 描 述 了 在 这 个 示例 中 将 要 创建 的 表 。 
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表 12-2 Entries 表 





























列 名 称 列 类 型 
entry_id 三 动 增长 的 正 整数 ， 非 空 
title 最 长 为 100 字 符 的 文本 
entry 任意 长 度 的 文本 
date_entered 包含 了 该 行 添加 日 期 和 时 间 的 时 间 戳 


在 这 个 例子 中 ， 将 要 创建 一 个 表 ， 用 于 存放 通过 HTML 表 单 提交 的 信息 。 在 12.6 节 中 ， 还 将 
编写 脚本 ， 将 提交 的 数据 插入 到 这 里 创建 的 表 中 。 


这 创建 一 个 新 表 


(1) 在 文本 编辑 器 或 IDE 中 打开 新 的 PHP 文 档 , 命名 为 create_table.php (参看 脚本 12-4) : 


<!IDOCTYPE html PUBLIC "“-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Create a Table</title> 

</head> 

<body> 


脚本 12-4 ”要 创建 数据 库 表 ， 首 先 定义 适当 的 SQL 语 句 ， 然 后 调用 mysql_query () 图 数 























1 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

对 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 

6 <title>Create a Table</title> 

7 </head> 

8 <body> 

9 ”<?php // 脚本 12-4 - create_table.php 

10 /* 脚本 连接 到 MYSQL 服务 器 。 选 中 数据 库 并 创建 数据 库 表 。*/ 

小 击 

12 // 连接 服务 器 并 选中 数据 库 : 

13 if ($dbc = @mysqgql connect('localhost', 'username', 'password')) { 

14 

15 // 如 果 数 据 库 无 法 选中 ， 打 印 错误 信息 : 

16 if (!@mysql_select db('myblog', $dbc)) { 

7 print '<p style="color: red;">Could not select the database because:<br/>' 
mysql_error($dbc) . '.</p>'; 

18 mysql_close(s$sdbc); 

49 $sdbc = FALSE; 

20 } 

2 

22 } else { // 连接 失败 。 

23 print '<p style="color: red;">Could not connect to MySQL:<br/>' . mysql error() . '.</p>'; 12 

24 } 

25 

26 if ($dbc) { 

27 


28 // 定义 查询 : 
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29 $query = 'CREATE TABLE entries ( 

30 entry_ id INT UNSIGNED NOT NULL AUTO INCREMENT PRIMARY KEY, 
31 title VARCHAR(100) NOT NULL, 

32 entry TEXT NOT NULL, 

33 date entered DATETIME NOT NULL 


34 )'; 

35 

36 // 执行 查询 : 

37 if (@mysql query ($query, $dbc)) { 

38 print '<p>The table has been created!</p>'; 

39 } else { 

40 print '<p style="color: red;"> Could not create the table because:<br/>' 
mysql error($dbc) . '.</p> <p>The query being run was: ' . $query . '</p>'; 

41 } 

42 

43 mysql_close ($dbc); // 关闭 连接 。 

44 

45 } 

46 ?> 


47 </body> 
48 </html> 


(2) 打开 一 个 PHP 代 码 片段 : 
<?php // 脚本 12-4 - create_table.php 
(3) 连接 到 MySQL 服 务 器 并 选中 数据 库 : 


if ($dbc = @mysql connect ('localhost', 'username', 'password')) { 
if (!@mysql_select db('myblog', $dbc)) { 
print '<p style="color: red;">Could not select the database because:<br/>' 
"mysql_error($dbc) . '.</p>'; 
mysql_close(sdbc); 
$sdbc = FALSE; 
} 
} else { 
print '<p style="color: red;">Could not connect to MySQL:<br/>' . mysql_ error() 
ey 
} 


这 是 前 面 脚本 中 使 用 的 代码 的 另 一 个 版 本 。 主 要 的 区 别 在 于 如 果 每 一 步 都 成 功 了 ， 则 不 会 打 
印 任何 消息 (这 里 我 们 假设 所 有 代码 都 能 正常 工作 )。 
如 果 出 于 某 些 原因 ， 无 法 连接 或 选中 数据 库 ， 则 会 打印 错误 信息 (参见 图 12-9)。 这 样 做 是 
意义 的 ， 因 为 如 果 数 据 库 无 法 选中 ， 就 不 能 尝试 在 其 中 创建 表 。 表 示 连 接 的 $dbc 变 量 会 被 设 
置 为 FALSE， 指 示 不 要 执行 赋予 其 的 CREATE 查 询 (参见 第 (4) 步 )。 






























































(4) 创建 查询 来 建 并 表 : 
i (SAbc)ot 
Squery = 'CREATE TABLE entries ( 














entry_id INT UNSIGNED NOT NULL AUTO_INCREMENT PRIMARY KEY, 
title VARCHAR(100) NOT NULL, 

entry TEXT NOT NULL, 
date_entered DATETIME NOT NULL 
jy 
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@ © (Sy Create a Table x \ a 


€3© © © phpvgs4:8888/create table.php 








Could not select the database because: 
Unknown database Imyblogs . 














图 12-9 根据 MySQL 错 误 消 ， 息 和 打印 出 来 的 正在 执行 的 查询 语句 ， 
应 该 能 够 找 出 导致 脚本 无 法 正确 执行 的 错误 


首先 ， 如 果 $qbc 具 有 值 ， 则 可 以 创建 表 ， 如 果 没 有 值 ， 则 意味 着 链接 无 法 建立 或 数据 库 无 
法 选中 , 这 样 之 后 的 代码 都 不 应 执行 。 对 于 查询 本 身 , 我 们 可 以 将 其 分 割 成 易于 识别 的 几 个 部 分 。 
首先 ， 要 创建 新 表 ， 应 写 下 CREATE TABLE tablename (其 中 tablename 应 该 用 实际 所 需 的 表 
名 称 琳 换 掉 )。 然 后 ， 在 括号 中 列 出 所 需 的 每 一 列 ， 列 之 间 用 逗号 分 开 。 表 和 列 的 名 字 应 该 是 字 
母 或 数字 ， 不 能 有 空格 。 

该 表 的 第 一 列 称 作 entry_ida， 这 是 一 个 无 符号 整数 (INT UNSIGNED， 表 示 它 只 能 存放 正 
整数 )。 通 过 添加 单词 NOT NULL， 可 以 指定 该 列 对 于 每 一 行 都 必须 具有 一 个 值 。 其 值 在 每 次 插入 
新 行 时 自动 增加 (AUTO INCREMENT)， 并 用 作 主 键 。 

接 下 来 的 两 列 由 文本 构成 。 其 中 一 个 名 为 title， 被 限制 为 100 字 符 。 第 二 个 名 为 entry， 
可 以 拥有 几乎 无 限 的 大 小 。 这 NULL ， 表 示 它 们 是 必需 的 字段 。 

最 后 ，aate_entereq 列 是 一 个 时 间 惟 ， 标 志 着 每 一 条 记录 是 何 时 被 添加 到 表 中 的 。 

(5) 执行 查询 : 


if (@mysql_query ($query, S$dbc)) { 
print '<p>The table has been created.</p>'; 























} else { 
print '<p style="color: red;">Could not create the table because:<br/>' 
—mysql_error($dbc) . '.</p><p>The query being run was: ' . Squery . '</p>'; 


} 

创建 表 ， 使 用 $query 变量 作为 参数 来 调用 mysql_query () 函数 。 如 果 发 生 错 误 ， 会 打印 
MySQL 错 误 以 及 $query 变量 的 值 。 最 后 这 一 步 (打印 正在 执行 的 查询 语句 ) 是 非常 有 用 的 调试 
技术 (参见 图 12-10)。 





Could not create the table because: 
CREATE command denied to user 'username'@ 'localhost' for table 
‘entries'. 


The query being run was: CREATE TABLE entries ( entry_id INT 
UNSIGNED NOT NULL AUTO_INCREMENT PRIMARY 
KEY ,title VARCHAR(100) NOT NULL., entry TEXT NOT 
NULL, date_entered DATETIME NOT NULL ) 














图 12-10 ”如 果 查 询 发 生 错误 ,会 报告 MySQL 错 误 ， 并 显示 出 错 的 查询 语句 (出 于 调试 目的 ) 
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(6) 关闭 数据 库 连 接 并 完成 $dbc 条 件 语句 : 


mysql_close(s$sdbc); 
} 


(7) 完成 PHP 代 码 和 HTML 页 面 : 


?> 
</body> 
</html> 


(8) 将 脚本 保存 为 create_table .pnp， 放 置 在 启用 了 PHP 的 服务 器 上 的 适当 目录 中 ， 并 在 
Web 浏 览 器 中 测试 (参见 图 12-11)。 


© 日 © (Wy Create a Table x \ + ) 


© © © phpvqs4:8888/create_table.php 




















The table has been created! 





图 12-11 如 果 一 切 正常 ， 则 会 看 到 该 信息 





ww 提示 

口 编写 SQL 查询 时 并 不 一 定 要 像 这 里 这 样 全 部 使 用 全 大 写字 母 ， 但 这 样 做 有 助 于 区 分 SQL 

关键 字 与 表 和 列 的 名 字 。 

D 在 大 型 Web 应 用 程序 中 ,强烈 建议 将 数据 库 连 接 和 选中 代码 (这 里 的 第 13 一 24 行 ) 放置 在 
单独 的 文件 中 , 并 放置 在 Web 目 录 之 外 。 然 后 ， 每 个 需要 查询 数据 库 的 页 面 都 来 包含 这 个 
外 部 文件 。 

口 只 要 查询 在 数据 库 上 运行 成 功 了 ，mysql_query () 函数 就 会 返回 TRUE。 但 这 并 不 意味 

着 期 望 的 结果 就 必定 发 生 了 。 

口 本 章 只 讲述 了 MySQL 和 SQL 相关 的 基本 知识 (包括 列 类 型 )。 当 对 这 些 基础 知识 掌握 之 后 ， 

可 能 希望 查看 其 他 资源 ， 这 些 在 附录 B 中 都 列 出 了 。 

口 一 般 不 需要 使 用 PHP 脚 本 创建 表 ， 就 像 一 般 不 需要 使 用 PHP 脚 本 创建 数据 库 一 样 。 但 是 当 
刚 开 始 使 用 MySQL 时 ， 这 是 完成 所 需 工 作 的 简单 方式 


12.6 ”向 数据 库 插 入 数据 


前 面 已 经 提 到 过 ， 该 数据 库 将 用 于 一 个 博客 ， 这 是 一 种 在 线 的 日 记 。 博 客 条 目 〈 由 标题 和 文 
本 构成 ) 将 通过 一 个 页 面 添 加 到 数据 库 中 ,然后 显示 在 男 一 个 页 面 中 。 这 是 一 个 简单 的 示例 ,但 
与 数据 库 的 使 用 关系 紧密 。 

在 12.5 节 中 ,我们 创建 了 一 个 表 , 它 由 4 列 构 成 :entry_id.、title、entry 和 date_entered。 
从 所 使 用 的 函数 来 看 ， 向 表 中 插入 信息 的 过 程 和 创建 表 的 过 程 是 类 似 的 , 但 是 用 的 SQL 查 询 是 不 
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同 的 。 要 插入 记录 ， 可 以 通过 下 面 这 两 种 模式 来 使 用 INSERT SQL 语法 : 


INSERT INTO tablename VALUES (valuel, value2, value3, etc.) 
INSERT INTO tablename (column1_ name, column2_ name) VALUES (valuel, value2) 


该 查询 以 INSERT INTO tablename 开 始 。 然后 可 以 指定 要 向 哪些 列 插 入 值 (也 可 以 不 指定 )。 
后 面 一 种 形式 更 为 明确 ， 因 此 我 也 更 加 推荐 这 种 形式 ， 但 当 插 入 大 量 列 时 ， 这 种 形式 更 加 元 长 。 
不 管 采 用 哪 种 形式 ， 必 须 为 表 中 的 每 一 行列 出 正确 数量 的 列 和 值 的 类 型 。 

值 放 在 括号 中 , 每 个 值 之 间 用 逗号 分 隔 。 字 符 串 和 日 期 这 样 的 非 数 字 值 需要 使 用 引号 ,数字 
值 则 不 用 : 

INSERT INTO example (name, age) VALUES ('Jonah', 1) 

使 用 mysql_aquery () 国 数 可 以 在 数据 库 上 运行 该 查询 。 由 于 INSERT 查 询 通常 都 很 复杂 ， 
此 有 必要 给 每 个 查询 都 赋予 一 个 变量 ， 然 后 将 该 变量 传递 给 mysql_query () 函数 (和 前 面 的 示 
例 一 样 )。 

作为 演示 ， 我 们 来 创建 一 个 页 面 将 blog 条 目 添加 到 数据 库 中 。 和 之 前 章节 中 的 例子 一 样 ， 该 
页 面 同时 显示 和 处 理 HTML 表 单 。 然 而 ， 在 进入 这 个 示例 之 前 ， 要 注意 该 脚本 包含 安全 汤 洞 , 我 
们 将 在 本 章 的 12.7 节 中 解释 并 修复 该 漏洞 。 

局 通过 HTML 表 单 将 数据 插入 到 数据 库 中 


(1) 在 文本 编辑 器 或 IDE 中 打开 一 个 新 的 PHP 文 档 , 命名 为 a9d_entry .php (参看 脚本 12-5 ) 。 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1l/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Add a Blog Entry</title> 

</head> 

<body> 

<hi>Add a Blog Entry</h1l> 























7 









































脚本 12-5 ”向 数据 库 中 添加 信息 的 查询 语句 非常 简单 ， 但 要 确保 括号 中 值 的 数量 和 数据 库 表 中 
列 的 数量 是 匹配 的 








和 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 <html xmlns="http://www.w3.o0rg/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>Add a Blog Entry</title> 

7 </head> 

8 <body> 

9 <h1l>Add a Blog Entry</h1i> 





10 <?php // 脚本 12-5 - add_entry.php 
11 /* This script adds a blog entry to the database. */ 


13 if ($_SERVER['REQUEST METHOD'] == 'POST') { // 处 理 表 单 。 
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14 

15 // 连接 服务 器 并 选中 数据 库 : 

16 $sdbc = mysql_ connect('localhost', 'username', 'password'); 

a mysql_select db('myblog', $dbc); 

18 

19 // 验证 表单 数据 : 

20 $sproblem = FALSE; 

2 if (!empty($_POSTI['title']) && !empty($_POST['entry'])) { 

2.2. stitle = trim(strip tags($_ POST['title'])); 

23 $sentry = trim(strip tags($_ POST['entry'])); 

24 } else { 

2.5 print '<p style="color: red;">Please submit both a title and an entry.</p>'; 

26 $sproblem = TRUE : 

223 } 

28 

29 IE (1Sproblem) { 

30 

31 // 定义 查询 : 

32 $query = "INSERT INTO entries (entry id, title, entry, date entered) VALUES 

(0, '$title', '$entry', NOW())"; 

33 

34 // 执行 查询 : 

35 if (@mysql query($query, $dbc)) { 

36 print '<p>The blog entry has been added!</p>'; 

37 } else { 

38 print '<p style="color: red;">Could not add the entry because:<br/>' . 
mysql_error ($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 

} 
} // 一 切 正 常 ! 


mysdql_close(Saqbc); // 关闭 连接 。 
} // 结束 提交 条 件 语句 。 


// 显示 表单 : 

2 

<form action="add entry.php" method="post"> 
<p>Entry Title: <input type="text" name="title" size="40" maxsize="100" /></p> 
<p>Entry Text: <textarea name="entry" cols="40" rows="5"></textarea></p> 
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52 <input type="submit" name="submit" value="Post This Entry!" /> 
53 </form> 
54 </body> 


55 </html> 
(2) 创建 初始 PHP 段 ， 并 检查 是 否 正 在 提交 表单 : 


<?php // 脚本 12-5 - add_entry .php 


























if (S$_SERVER['REQUEST METHOD'] == 'POST') { 
(3) 连接 并 选中 数据 库 : 
$sdbc = mysql_ connect('localhost', 'username', 'password'); 


mysql_select db('myblog', $dbc); 
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此 时 ,如 果 正 在 按 顺 序 运行 这 些 示 例 ， 本 书 将 假设 已 经 有 了 能 够 工作 的 连接 和 选中 过 程 ， 因 
此 省 略 了 所 有 的 条 件 语句 和 错误 报告 ( 极 大 地 缩短 了 脚本 )。 如 果 在 连接 和 选中 数据 库 时 遇 到 了 
问题 ， 请 使 用 本 章 之 前 列 出 的 代码 。 
(4) 验证 表单 数据 : 
$sproblem = FALSE; 
if (!empty($_POST['title']) && !lempty($_POST['entry'])) { 
$title = trim(strip tags ($_POST['title'])); 
Sentry = trim(strip_ tags ($_POST['entry'])); 
} else { 
print '<p style="color: red;">Please submit both a title and an entry.</p>'; 


sproblem = TRUE; 


在 INSERT 查 询 中 使 用 表单 数据 之 前 ， 应 该 对 其 进行 验证 。 这 里 只 使 用 了 最 少 的 验证 ， 确 保 
确实 提供 了 某 些 值 。 如 果 验 证 成 功 ， 这 些 值 会 在 截 去 两 端的 空格 后 ， 赋 给 新 的 变量 。 如 果 没 有 通 
过 验证 ， 会 打印 错误 消息 (参见 图 12-12) 并 将 $Sproblem 标 志 变量 设置 为 TRUE (因为 发 生 了 错 
误 )。 


























Add a Blog Entry 


Please submit both a title and an entry. 


Entry Title: 














Entry Text: < 
图 12-12 PHP 还 是 执行 了 一 些 基 本 的 表单 验证 ， 因 此 空 的 记录 是 不 会 插入 到 数据 库 中 的 
(5) 定义 INSERT 查 询 : 


if (!$Sproblem) { 
$query = "INSERT INTO entries (entry_id, title, entry, date entered) VALUES 
(0, 'S$title', '$entry', NOW())"; 


该 查询 由 必需 的 INSERT INTO tablename 代 码 开始 。 然 后 列 出 了 与 提交 的 值 对 应 的 列 。 之 
后 是 单词 VALUES， 后 跟 4 个 值 (每 个 列 一 个 ， 按 顺序 放置 ) 放置 在 单 引 号 中 并 用 逗号 分 开 。 当 把 
该 查询 赋值 给 $query 变量 时 ， 要 使 用 双 引 号 这 样 变量 的 值 就 会 被 PHP 自 动 插入 。S$title 和 
s$entry 变 量 是 字符 串 ， 因 此 在 查询 中 要 用 引号 括 起 来 。 1 
由 于 entry_iq 列 被 设置 为 AUTO_INCREMENT， 因 此 可 以 使 用 0 作为 其 值 ，MySQL 会 自动 为 
该 列 使 用 逻辑 上 的 下 一 个 值 。 要 设置 date_entered 列 的 值 ， 可 以 使 用 MySQL 函 数 Now () 。 它 会 
将 当前 时 间作 为 值 插入 。 
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(6) 在 数据 库 上 运行 查询 : 


if (@mysql_ query ($query, $dbc)) { 
print '<p>The blog entry has been added!</p>'; 





} else { 
print '<p style="color: red;"> Could not add the entry because:<br/>' . mysqgl_ 
>error($dbc) . '.</p><p>The query being run was: ' . Squery . '</p>'; 


} 


一 旦 定义 好 查询 , 就 可 以 使 用 mysql_query () 国 数 运行 它 。 将 对 该 函数 的 调用 作为 if-else 
语句 的 条 件 ， 通 过 这 种 方式 可 以 基于 查询 执行 的 结果 打印 简单 的 消息 。 

由 于 这 本 身 就 是 一 个 调试 工具 ， 所 以 当 查 询 没 有 正确 运行 时 ， 会 在 Web 浏 览 器 中 打印 出 
MySQL 错 误 和 正在 运行 的 查询 (参见 图 12-13)。 


Add a Blog Entry 


Could not add the entry because: 

You have an error in your SQL syntax; check the manual that corresponds to your 
MySQL server version for the right syntax to Use near 's January 1st, 2011, the first 
day of six weeks of writing the date incorrectly. at line 1. 





The query being run was: INSERT INTO entries (entry_id, title, entry, date_entered) 
VALUES (0,'Happy New Year! ,Its January 1st, 2011, the first day of six weeks of 
writing the date incorrectly.',, NOWO) 














图 12-13 ”如 果 INSERT 查 询 不 能 工作 ， 会 打印 出 MySQL 错 误 和 正在 运行 的 查询 
(7) 关闭 Sproblem 条 件 语 句 、 数 据 库 连接 ， 完 成 主 条 件 语句 和 PHP 片 段 : 


}】 // 一 切 正常 ! 
mysql_close(s$dbc); 
} // 结束 提交 条 件 语 向 。 


从 这 里 开始 ， 表 单 将 会 被 显示 出 来 。 
(8) 创建 表单 : 





<form action="add_ entry.php" method="post"> 
<p>Entry Title: <input type="text" name="title" size="40" maxsize="100" /></p> 
<p>Entry Text: <textarea name="entry" cols="40" rows="5"></textarea></p> 
<input type="submit" name="submit" value="Post This Entry!" /> 

</form> 


这 个 HTML 表 单 非常 简单 ， 用 于 请 求 博客 的 标题 和 内 容 。 一 个 很 好 的 经 验 规则 是 ， 为 表单 的 
输入 框 使 用 的 名 字 应 该 和 数据 库 中 的 列 名 字 对 应 起 来 。 这 样 做 可 以 降低 错误 发 生 的 几率 
(9) 完成 HTML 页 面 : 


</body> 
</htmil> 


(10) 将 脚本 保存 为 a9dq_entry .php， 放 置 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 ， 然 后 在 
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Wepb 浏 览 器 中 测试 (参见 图 12-14 和 图 12-15)。 


| @O > ) QAdda Blog Entry x \ a 


€ > | 图 phpvqs4:8888/add_entry.php 















































Add a Blog Entry 
Entry Title: |Happy New Year! 
wi th date incorrecty, oY of SH Wee oF Add a Blog Entry 
Entry Text: 4 The blog entry has been added! 
Entry Title: 
图 12-14 用 于 向 数据 库 中 添加 条 目的 表单 图 12-15 “如果 INSERT 查 询 正确 地 运行 了 ， 会 打印 









































一 条 消息 并 再 次 显示 表单 
在 表单 的 值 中 应 该 避免 使 用 撒 号 , 否则 将 会 看 到 如 图 12-13 所 示 的 结果 。12.7 市 会 详细 介绍 此 
问题 并 给 出 解决 方法 。 
w 提示 
口 MySQL 支 持 使 用 下 面 的 格式 一 次 插入 多 条 记录 : 


INSERT INTO tablename (column1 name, column2_ name) VALUES (valuel, value2), 
> (value3, value4); 


然而 ， 大 多 数 其 他 数据 库 应 用 程序 不 支持 这 种 结构 。 
口 要 检索 为 AUTO_INCREMENT 列 自动 产生 的 数值 ， 可 以 使 用 mysql_insert_id() 国 数 。 
口 由 于 主键 自动 递增 ， 查 询 也 可 以 写成 : 


INSERT INTO entries (title, entry, date_ entered) VALUES ('s$title', '$entry', NOW())"; 





























创建 示例 
UUOU0O00000PHEU MySQIUUOUOOOUOOOUO0OUSQUOUOOO0OO0OO00 
加 
国 
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12.7 安全 查询 数据 下 


在 介绍 之 前 的 步骤 时 ， 本 书 已 经 提 到 过 , 这 里 编写 的 代码 中 存在 着 很 恶劣 的 安全 漏洞 。 对 于 
这 样 的 代码 ， 如 果 某 些 用 户 提 交 的 文本 中 包含 一 个 撒 号 ,这样 的 数据 就 会 破坏 SQL 查 询 (参见 图 
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12-16) 。 这 个 结果 很 明显 不 是 我 们 想 要 的 ， 但 是 为 什么 这 是 不 安全 的 呢 ? 
如 果 一 个 恶意 用 户 知道 他 可 以 通过 输入 一 个 撒 号 来 破坏 查询 , 那么 他 就 可 以 试图 利用 该 漏洞 
运行 他 自己 的 查询 。 如 果 有 的 用 户 提 交 ' ;DROP TABLE entries; 作 为 blog 条 目的 标题 ， 则 产生 























的 查询 将 是 : 
INSERT INTO entries (entry_id, title, entry, date entered) VALUES 
(0, '';DROP TABLE entries;', '<entry text>', NOW()) 
Add a Blog Entry 
Could not add the entry because: 
You have an error in your SQL syntax: check the manual that corresponds to your 
MySQL server version for the right syntax to use near :DROP TABLE entries;', ", 
NOWO) at line 1. 
The query being run was: INSERT INTO entries (entry_id, title, entry, date_entered) 
VALUES (0,";DROP TABLE entries;, ", NOW')) 
图 12-16 ”连接 时 的 撤 号 会 破坏 查询 ， 因 为 撤 号 (或 单 引号 ) 用 于 分 隔 查 询 中 使 用 的 字符 串 


























第 一 个 撒 号 完成 了 查询 中 博客 标题 的 值 。 接 下 来 ， 分 号 结束 了 INSERT 查 询 语句 。 这 导致 原 
本 的 查询 语句 发 生 语法 错误 。 接 下 来 ， 向 数据 库 提交 了 第 二 个 查询 语句 一 一 DROP TABLE 
entries， 这 个 语句 顺利 执行 ， 而 原来 的 INSERT 查 询 会 失败 。 这 被 称 作 SQL 注 入 攻击 (SQL 
injection attack) ， 但 幸运 的 是 这 很 容易 避免 。 

要 避免 这 样 的 问题 ， 在 向 查询 传递 可 能 不 安全 的 数据 时 ， 请 使 用 mysql_real_escape_ 
string () 函数 。 该 函数 会 对 任何 可 能 危险 的 字符 进行 转 义 ， 即 在 其 前 面 添加 一 个 反 斜 线 ， 确 保 
查询 中 使 用 的 数据 是 安全 的 : 

$var = mysql_real_ escape _ string($var, $dbc); 


下 面 我 们 对 前 面 的 脚本 应 用 该 函数 。 


这 使 查询 数据 更 安全 


























(1) 如 果 还 没 打 开 add_entry .php 脚 本 (参看 脚本 12-5)， 在 文本 编辑 器 或 IDE 中 打开 它 。 
(2) 更 新 所 指派 的 Stitle 和 Sentry 变 量 以 便 读 取 (参看 脚本 12-6): 


stitle 
$sentry 


mysql_real _ escape _ string(trim(strip tags($_POST['title'])), $dbc); 
mysql_real_ escape_ string(trim(strip_ tags($_POST['entry'])), $dbc); 





脚本 12-6 要 使 Web 应 用 程序 和 数据 库 更 为 安全 ， 需 要 针对 在 查询 中 使 用 的 表单 数据 应 用 


mysql_real_escape_string () 水 数 





<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.o0org/1999/xhtml" xml:lang="en" lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Add a Blog Entry</title> 


OOPRODP 
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</head> 

<body> 

<hil>Add a Blog Entry</hi1l> 

<?php // 脚本 12-6 - adgd_entry.php #2 

/* 脚本 向 数据 库 中 添加 一 篇 博客 ， 并 使 查询 更 加 安全 。*/ 





if (isset($_POST['submitted'])) { // 处 理 表单 。 


// 连接 服务 器 并 选中 数据 库 : 
$sdbc = mysql_ connect('localhost', 'username', 'password'); 
mysql_select_ db('myblog', S$dbc); 


// 验证 表单 数据 并 确保 安全 : 
$sproblem = FALSE; 
if (!empty($_POST['title']) && !lempty($_POST['entry'])) { 
$title =mysql real escape string(trim(strip tags($ POST['title'])), $dbc); 
$entry = mysql real escape string(trim(strip tags($ POST['entry'])), $dbc); 
} else { 
print '<p style="color: red;">Please submit both a title and an entry.</p>'; 
sproblem = TRUE; 





if (!S$problem) { 








// 定义 查询 : 
Squery = "INSERT INTO entries (entry_id, title, entry, date_ entered) VALUES 
(0, '$title', '$entry', NOW())"; 


// 执行 查询 : 
if (@mysql_ query ($query, S$dbc)) { 
print '<p>The blog entry has been added!</p>'; 


} else { 
print '<p style="color: red;">Could not add the entry because:<br/>' 
mysql_error ($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 
} 
} // 一 切 正 常 ! 


mysql_close($dbc); // 关闭 连接 。 
}】 // 结束 提交 条 件 语句 。 


// 显示 表单 : 

?> 

<form action="add entry.php" method="post"> 
<p>Entry Title: <input type="text" name="title" size="40" maxsize="100" /></p> 
<p>Entry Text: <textarea name="entry" cols="40" rows="5"></textarea></p> 
<input type="submit" name="submit" value="Post This Entry!" /> 
<input type="hidden" name="submitted" value="true" /> 

</form> 

</body> 

</html> 


这 两 行 会 大 大 改善 脚本 的 安全 性 和 功能 。 提 交 的 两 个 变量 ,首先 截 去 了 两 端的 空格 ,然后 被 
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传递 给 了 mysql_real_escape_string()。 其 结果 可 以 安全 地 用 于 查询 。 

如 果 你 觉得 上 面 的 代码 (将 3 个 函数 的 返回 值 赋 给 同一 个 变量 ) 不 太 容 易 理解 ， 可 以 将 代码 
分 开 写 成 3 条 语句 : 

stitle $_POST['title']; 

$title trim(strip_ tags ($title)); 
$title = mysql real escape_ string ($title, $dbc); 
(3) 保存 脚本 , 将 其 放置 在 启用 了 PHP 的 服务 器 上 的 适当 目录 中 ,然后 在 Web 训 览 器 中 测试 它 
(参见 图 12-17 和 图 12-18 ) 。 


Add a Blog Entry 


Entry Title: its Another Testl 














“Will these quotes and apostrophes cause problems? ， 
you ask. | don't think so! 















































a Add a Blog Entry 
The blog entry has been added! 
图 12-17 ”现在 再 在 表单 中 输入 撒 号 图 12-18 已 经 不 会 产生 问题 了 
V 提示 
口 (在 本 章 稍 后 的 部 分 ) 如 果 看 到 显示 出 来 的 博客 条 目 在 撤 号 之 前 有 额外 的 反 斜 线 ， 则 很 可 








能 是 使 用 了 PHP 版 本 6 之 前 的 版 本 ， 并 启用 了 Magic Quotes。( 即 便 没 有 使 用 mysal_ 
real_escape_string() ，Magic Quotes 也 会 自动 转 义 表单 数据 中 有 问题 的 字符 。) 如 果 
是 这 种 情况 ， 需 要 使 用 stripslashes () 函数 从 提交 的 值 中 移 除 额外 的 反 斜 线 : 


$title = mysql real escape string(stripslashes (trim(strip tags($ POST['title']))), $dbc); 


显示 MySQL 错 误 
0U0U MysSQLUOUOOOOODOSQUOUOODOD MySQLI OUUmysgl cuery(OUUOOOD 
回国 国医 SS 全 加 加 本 
国 
D000000000000000000000000000000000000MySQL 
国生 ggggggoowog 


12.8 ”从 数据 库 中 检索 数据 


本 章 要 演示 的 操作 数据 库 的 下 一 个 步骤 是 从 已 有 的 表 中 检索 数据 。 这 里 依然 使 用 mysql_ 
query () 国 数 ， 但 检索 数据 和 插入 数据 有 些 不 同 : 必须 将 检索 到 的 信息 赋 给 一 个 变量 ,然后 使 用 
另 一 个 国 数 获取 数据 。 
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检索 数据 的 基本 语法 是 SELECT 查询 : 
SELECT what columns FROM what table 
从 一 个 表 中 读 取 数据 ， 最 简单 的 查询 是 : 
SELECT * FROM tablename 
其 中 的 星 号 表示 查询 所 有 列 。 如 果 只 需要 返回 某 几 列 ， 可 以 对 查询 作出 限定 ， 如 : 
SELECT name, email FROM users 
这 个 查询 要 求 只 生成 两 列 (name 和 emai1l) 信息 。 要 记 住 , 这 种 结构 不 会 限制 返回 哪些 行 (或 
记录 )， 只 是 返回 所 有 行 的 某 些 列 。 
该 查询 的 另 一 种 修改 方式 是 添加 一 个 条 件 ， 限 第 
SELECT * FROM users WHERE name='Larry 
这 里 需要 的 是 表 中 所 有 列 的 信息 ， 但 只 返回 那些 name 列 等 于 Larry 的 行 。 
这 是 一 个 很 好 的 示例 ， 当 SQL 只 需要 少量 条 目 时 ， 这 样 做 很 有 效 而 且 很 灵活 。 
从 数据 库 中 检索 数据 和 向 数据 库 中 插入 数据 的 主要 区 别 在 于 ， 需 要 对 查询 进行 不 同 的 处 理 。 
需要 将 查询 的 结果 赋 给 一 个 变量 ， 
$result = mysql_query ($query, S$dbc); 
$dbc 是 一 个 打开 的 数据 库 连 接 的 引用 , 而 $result 是 一 个 查询 结果 集 的 引用 。$result 变 量 
接 下 来 会 提交 给 mysdl1_fetch_arravy() 国 数 ， 以 获取 查询 结果 ; 
Srow = mysql_fetch array ($result); 
国 数 从 结果 集中 每 次 获取 一 行 数据 , 并 在 这 一 过 程 中 创建 一 个 数组 。 这 个 数组 会 使 用 已 选择 
的 列 名 作为 索引 : Srow['name']0 srow['email'] 等 。 和 使 用 其 他 数组 一 样 ， 必 须 通 过 数据 
库 中 定义 的 确切 列 名 来 引用 其 中 的 列 〈 列 名 是 区 分 大 小 写 的 )。 因 此 ， 在 这 个 例子 中 ， 必 须 使 用 
srow['email']， 而 不 能 使 用 Srow['Email']。 
如 果 查 询 会 返回 多 行 数据 ， 在 循环 中 执行 nysql_fetch_array () 国 数 ， 以 获取 所 有 的 行 : 


while ($row = mysql_ fetch array ($result)) { 
// Do something with Srow . 




















返回 哪些 行 ， 这 通过 WHERE 子 句 来 完成 : 















































} 

在 循环 的 每 次 从 代 中 ,来 自 查 询 结 果 (通过 $result 引用 ) 的 下 一 行 信息 会 被 转移 到 一 个 称 
作 $row 的 数组 中 。 这 一 过 程 一 直 持 续 ， 直 到 从 返回 结果 中 再 也 找 不 到 任何 行 信 息 为 止 。 在 循环 
内 部 ， 可 以 对 $row 执行 任何 操作 。 

了 解 这 个 系统 的 最 佳 方 式 就 是 去 尝试 它 。 接 下 来 我 们 将 编写 一 个 脚本 , 检索 存放 在 entries 
表 中 的 条 目 并 显示 它们 (参见 图 12-19)。 之 前 应 该 已 经 创建 了 这 个 表 ， 并 且 不 止 一 次 地 运行 了 
add_entry .php。 
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My Blog 
It's Another Test! 


"Will these quotes and apostrophes cause problems?", you 
ask. I don't think so! 
Edit Delete 





Happy New Year! 


Today is January 1st, 2011, the first day of six weeks of 
writing the date incorrectly. 
Edit Delete 

















图 12-19 ”这 个 动态 Web 页 面 使 用 PHP 从 数据 库 中 提取 数据 








过 从 表 中 检索 数据 











(1) 在 文本 编辑 器 或 IDE 中 创下 
12-7): 


[en 


一 个 新 的 PHP 文 档 ， 命 名 为 view_entries .php (参看 脚本 





<!1DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>View My Blog</title> 
</head> 
<body> 
<hil>My Blog</h1> 











脚本 12-7 用 于 从 一 个 表 中 检索 所 有 数据 的 SQL 查 询 非 常 简单 ， 但 为 了 让 PHP 能 够 访问 返回 的 
每 一 行 记录 ， 需 要 在 循环 中 每 次 访问 结果 中 的 一 行 





工 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
6 <title>View My Blog</title> 

7 </head> 

8 <body> 

9 <hl>My Blog</h1> 

10 <?php // 脚本 12-7 - view_entries.php 

11 /* This Script retrieves blog entries from the database. */ 

12 

13 // 连接 服务 器 并 选中 数据 库 : 

14 sdbc = mysql connect('localhost', 'username', 'password'); 

15 mysql_select db('myblog', $dbc); 


Ea 
oO 
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17 // 定义 查询 : 


18 Squery = 'SELECT * FROM entries ORDER BY date entered DESC'; 

19 

20 if (Sr = mysql_query ($query，S$Sdbc)) { // 运行 查询 。 

21 

22 // 打印 返回 值 : 

23 while ($row = mysql fetch array($r)) { 

24 print "<p><h3>{$row['title']}</h3> 

25 {$row['entry']}<br/> 

26 <a href=\"edit entry.php?id={$row['entry id']}\">Edit</a> 

27 <a href=\"delete entry.php?id={$row['entry id']}\">Delete</a> 

28 </p><hr />\n"; 

29 } 

30 

31 } else { // 没有 运行 查询 。 

32 print '<p style="color: red;">Could not retrieve the data because:<br/>' 
mysql_error($dbc) . '.</p><p>The query being run was: ' . Squery . '</p>'; 

33 ”】 // 结束 查询 条 件 语句 。 

34 

35 mysql_close($dbc); // 关闭 连接 。 

36 

37 ?> 

38 </body> 


39 </html> 
(2) 定义 PHP 片 段 并 连接 到 数据 库 : 


<?php // 脚本 12-7 - view_entries.php 
$sdbc = mysql_ connect('localhost', 'username', 'password'); 
mysql_select_ db('myblog', S$dbc); 


(3) 定义 SELECT 查询 : 


Squery = 'SELECT * FROM entries ORDER BY dqate_enteredq DESC'; 


该 基本 查询 告诉 数据 库 需 要 查看 entries 表 中 所 有 行 所 有 列 的 数据 。 返回 的 结果 应 该 进行 排 
序 ， 这 通过 ORDER BY 子 句 指定 了 按照 输入 的 时 间 (记录 在 date_entered 列 中 ) 进行 排序 ， 最 
先 显示 最 近 插 入 的 记录 。 语 名 最 后 的 pEsc 选 项 是 descending 的 简写 。 如 果 将 查询 改 为 ORDER BY 
date_entered ASC， 则 最 新 插入 的 记录 将 被 排 在 最 后 。 

(4) 运行 查询 : 

if (Sr = mysql query ($query, S$dbc)) { 

SELECT 查询 和 其 他 查询 的 运行 方式 一 样 。 然 而 ， 查 询 的 结果 需要 赋 给 Sresult (或 者 更 短 
，S$r) 变量 ， 以 供 之 后 引用 。 

(5) 打印 返回 的 值 : 


while ($row = mysql_ fetch array ($sr)) { 
print "<p><h3>{$row['title']}</h3> 
{Srow['entry']}<br/> 
<a href=\"edit_ entry.php?id={$row['entry_id']}\">Edit</a> 
<a href=\"delete _ entry.php? id={$row['entry_id']}\"> Delete</a> 
D> NT 














| 发 
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该 循环 将 包含 有 $r 中 返回 的 第 一 条 记录 的 数组 赋 给 $row 变量 。 然 后 循环 会 执行 下 面 的 命令 
(print 语 句 )。 每 当 循环 回 到 起 点 ,如果 还 有 其 他 行 的 话 ， 它 会 再 次 将 下 一 行 赋值 给 $row。 如 此 
继续 下 去 ， 直 到 无 法 再 获取 更 多 行 信 息 为 止 。 

数组 的 键 就 是 表 中 列 的 名 字 ， 也 就 是 entry_id、title 和 entry (这 里 无 需 打 印 出 daate_ 
entered)。 

在 每 个 条 目的 底部 , 还 创建 了 两 个 链接 : 指向 edit_entry.php 和 delete_entry.php。 这 
些 功能 将 在 本 章 其 他 部 分 中 编写 。 每 个 链接 都 将 条 目 在 数据 库 中 的 ID 值 传递 到 URL 中 。 在 另外 两 
个 页 面 对 blog 条 目 进行 编辑 和 删除 时 ， 该 信息 是 必 不 可 少 的 。 

(6) 处 理 查 询 无 法 执行 的 错误 : 

}) else { // 没有 运行 查询 。 

print '<p style="color: red;">Could not retrieve the data because:<br/>' . 


"mysql_error($dbc) . '.</p> <p>The query being run was: ' . Squery . '</p>'; 
】 // 结束 查询 条 件 语句 。 


如 果 无 法 在 数据 库 上 运行 查询 ， 则 应 该 打印 出 该 查询 ， 以 及 MySQL 错 误 (用 于 调试 )。 
(7) 关闭 数据 库 连 接 : 
mysql_close(sdbc); 


(8) 完成 PHP 片 段 和 HTML 页 面 : 


?> 
</body> 
</html> 


(9) 将 脚本 保存 为 view_blog .php, 放置 在 启用 了 PHP 的 服务 器 上 的 适当 目录 中 , 然后 在 Web 
浏览 器 中 进行 测试 (参见 图 12-19)。 

(10) 如 果 需 要 的 话 ， 可 以 使 用 adg_entry .php 页 (参看 脚本 12-6) 添加 另 一 条 博客 记录 ,并 
再 次 运行 该 页 面 (参见 图 12-20)。 


My Blog 


This is the newest post! 





This is so absolutely amazing that Tm downright speechless! 
Edit Delete 





It's Another Test! 


"Will these quotes and apostrophes cause problems?", you ask. I don't 
think so! 
Edit Delete 








Happy New Year! 




















图 12-20 ”感谢 SELECT 查 询 ， 将 返回 的 记录 按照 输入 的 结果 进行 了 排序 ， 
使 最 新 添加 的 记录 总 是 列 在 最 前 面 
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(11) 如 果 希 望 的话 ， 检 查 页 面 的 源 代码 ， 查 看 动态 生成 的 链接 (参见 图 12-21)。 





<hl>My Blog</h1> 
<p><h3>This is the newest post!</h3> 
This is so absolutely amazing that I'm dowr 
<a href="edit entry.php?id=4">Edit</a> 
<a href="delete entry.php?id=4">Delete</a> 
</p><hr /> 
<p><h3>It's Another Test!</h3> 
"Will these quotes and apostrophes cause pr 
so!l<br /> 
<a href="edit entry.php?id=3">Edit</a> 
<a href="delete entry.php?id=3">Delete</a> 
</p><hr /> 
<p><h3>Happy New Year!</h3> 
Today is January lst, 201]1, the first day < 
incorrectly.<br /> 
<a href="edit entry.php?id=2">Edit</a> 
<a href="delete entry.php?id=2">Delete</a> 
</p><hr /> 














图 12-21 ”页面 HTML 源 代码 的 一 部 分 。 注 意 ， 两 个 链接 都 在 URL 后 面 追 加 了 ?id=x 


Vv 提示 

口 mysql_fetch_array () 国 数 还 可 以 接受 另 一 个 参数 , 这 是 一 个 常量 , 它 指定 了 需要 返回 

哪 种 类 型 的 数组 。MYSQL_Assoc 返 回 关 联 数组 ， 而 MYsQL_NUM 返 回 按 数值 索引 的 数组 。 

口 mysal_num_rows () 国 数 返回 由 SELECT 查询 返回 的 记录 行 数 。 

口 还 可 以 分 页 返回 数据 ， 如 每 页 返回 10 条 或 20 条 记录 (就 像 Google 那 样 )。 但 这 样 做 需要 编 
写 更 多 更 高 级 代码 ， 这 已 经 超出 了 本 书 的 讨论 范围 。 请 参考 我 写 的 《PHP6 与 MYSQL 基础 
教程 》， 或 在 线 查 找 一 些 代码 示例 和 教程 。 


12.9 删除 数据 库 中 的 数据 


有 的 时 候 可 能 需要 在 数据 库 上 运行 DELETE 查 询 。 该 查询 用 于 从 数据 库 中 移 除 记 录 。DELETE 
查询 的 语法 是 : 

DELETE FROM tablename WHERE column=value 

其 中 的 wHERE 子 句 并 不 是 必需 的 ， 但 如 果 省 略 它 ， 将 会 从 表 中 移 除 所 有 记录 。 还 要 记得 ， 一 
且 删 除了 一 条 记录 ， 就 无 法 恢复 它 〈 除 非 拥 有 数据 库 的 备份 ) 。 

作为 防范 措施 ， 如 果 你 希望 从 表 中 只 删除 一 条 记录 ， 就 可 以 为 查询 指定 LIMIT 子 句 : 

DELETE FROM tablename WHERE column=value LIMIT 1 

该 子 句 确保 了 最 多 只 有 一 条 记录 会 被 删除 。 一 旦 定义 好 查询 ， 就 可 以 再 次 使 用 mysql_ 
query () 国 数 来 执行 该 查询 ， 这 和 其 他 查询 一 样 。 

要 检查 DELETE 查 询 的 执行 效果 ， 可 以 使 用 mysql_affected_rows () 函数 。 该 函数 用 于 返 
回 被 INSERT、DELETE 或 UPDATE 查 询 影 响 到 的 行 数 。 

















ed 
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作为 示例 ， 我 们 将 编写 aelete_entry.php 脚 本 ，view_plog.php 页 面 中 的 链接 会 指向 该 





页 面 。 该 页 面 会 接收 URL 中 的 数据 库 记 录 id。 然 后 它 会 显示 这 个 条 目 , 确认 用 户 想 要 删除 它 ( 参 
见 图 12-22)。 如 果 用 户 单 击 了 按钮 ， 该 记录 将 被 删除 (参见 图 12-23)。 


学 








No Delete a Blog Entry 








€ 3 © © phpvqs4:8888/delete_entry.php?id=3 





Delete an Entry 
Are you sure you want to delete this entry? 
It's Another Test! 


"Will these quotes and apostrophes cause problems?", you ask. I don't think so! 
( Delete this Entry! ) 





@NON 


(Wy Delete a Blog Entry 


€ 3 © © phpvqs4:8888/delete_entry.php 








Delete an Entry 


The blog entry has been deleted. 








图 12-22 ” 当 用 户 进入 该 页 面 时 , 会 显示 出 博客 条 目 ， 




















用 户 必须 确认 他 们 想 删 除 它 


党 从 数据 库 中 删除 数据 


这 样 的 结果 


图 12-23 ”如果 DELETE 查 询 正确 执行 了 ， 用 户 会 看 到 














(1) 在 文本 编辑 器 或 IDE 中 打开 一 个 新 的 PHP 文 档 ， 命 名 为 aelete_entry.php (参看 脚本 


12-8 ) : 








<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 





<head> 


<meta http-equiv="content-type" content="text/html; 





<title>Delete a Blog Entry </title> 
</head> 
<body> 
<hil>Delete an Entry</h1> 








charset=utf-8" /> 


脚本 12-8 该 DELETE SQL 命 令 从 表 中 永久 地 删除 一 条 记录 (或 多 条 记录 ) 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 


1 

2 

3 

4 <head> 

5 <meta http-equiv="content-type" 
6 

学 

8 





</head> 
<body> 
9 <hil>Delete an Entry</hi1> 
10 <?php // 脚本 12-8 - delete_entry.php 
11 /* 脚本 删除 一 篇 博客 。*/ 
二 分 
13 // 连接 服务 器 并 选中 数据 库 : 





14 S$dbc = mysql_ connect('localhost', 'username', 


15 mysql_select db('myblog', S$dbc); 
16 


content="text/html; 


<title>Delete a Blog Entry</title> 


"http://www.w3.org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 


<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 


charset=utf-8" /> 


'password'); 
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17 if (isset($_GET['id']) && is numeric($_GET['id']) ) { // 在 表单 中 显示 条 目 : 

18 

19 // 定义 查询 : 

2.0 Squery = "SELECT title, entry FROM entries WHERE entry_id={$_GET['id']}"; 

21 if (Sr = mysql_query($Squery，S$dbc)) { // 运行 查询 。 

22 

23 Srow = mysql_fetch array($r); // 返回 信息 。 

24 

25 // 创建 表单 : 

26 print '<form action="delete entry.php" method="post"> 

27 <p>Are you sure you want to delete this entry?</p> 

28 <Do<h3sl -Orow[l title,) s, </h3 

29 srow['entry'] . '<br/> 

30 <input type="hidden" name="id" value="' . $_GET['id'] . '" /> 

3 <input type="submit" name="submit" value="Delete this Entry!" /></p> 

32 </form>'; 

33 

34 】 else { // 无 法 获取 信息 。 

记号 print '<p style="color: red; ">Could not retrieve the blog entry because:<br/>' 
mysql_error($dbc) . '.</p><p>The query being run was: ' . Squery . '</p>'; 

36 } 

3 了 

38 } elseif (isset($_POST['id']) && is_numeric($_POST['id'])) { // 处 理 表单 。 

39 

40 // 定义 查询 : 

41 $query = "DELETE FROM entries WHERE entry id={$ POST['id']} LIMIT 1"7 

42 Sr = mysql query($query, $dbc); // Execute the query. 

43 

44 // 检查 查询 结果 : 

45 if (mysql affected rows($dbc) == 1) { 

46 print '<p>The blog entry has been deleted.</p>'; 

47 } else { 

48 print '<p style="color: red;">Could not delete the blog entry because: <br/>' . 
mysql error($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 

49 } 

50 

51 } else { // 没有 获取 id。 

52 print '<p style="color: red;">This page has been accessed in error.</p>'; 

53 } // 结束 主 条 件 语句 。 

54 

55 mysql_close($dbc); // 关闭 连接 。 

56 

B71 3 

58 </body> 


59 </html> 


(2) 打开 PHP 代 码 并 连接 到 数据 库 : 


<?php // 脚本 12-8 - delete_entry.php 
$sdbc = mysql_ connect('localhost', 'username', 'password'); 
mysql_select db('myblog', S$dbc); 


(3) 如 果 页 面 从 URL 中 接收 到 了 有 效 的 条 目 ida， 则 定义 并 执行 一 个 SELECT 查询 : 


if (isset($_GET['id']) && is_numeric(S_GET['id']) ) { 
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"SELECT title, entry FROM entries WHERE entry_id={$_GET['id']}"; 
mysql_query ($query, $dbc)) { 











$query 
和 


要 显示 blog 条 目 ， 页 面 必须 确认 从 URL 中 接收 到 了 一 个 数值 ta。 由 于 该 1q 来 自 于 URL ( 当 用 
户 单 击 view_blog.php 中 的 链接 时 ， 参 见 图 12-24) ， 因 此 通过 $_GET['ia'] 进 行 引 用 。 

















Wo Delete a Blog Entry 


€ 3 © phpvqs4:8888/delete_entry.php 





Delete an Entry 


This page has been accessed in error. 











图 12-24 ”如 果 脚 本 没有 接收 到 ORL 中 的 ia 值 ， 会 打印 此 错误 信息 


这 个 查询 和 前 面 例子 中 使 用 的 SELECT 查 询 类 似 ， 区 别 在 于 这 里 添加 了 WHERE 子 句 ， 只 获取 
一 条 特定 的 记录 。 另 外 ， 由 于 只 有 title 和 entry 这 两 个 存储 值 是 必需 的 ， 所 以 这 里 只 选中 了 这 
两 列 。 

然后 使 用 mysql_query () 函数 在 数据 库 上 运行 该 查询 。 

(4) 检索 记录 ， 并 在 表单 中 显示 条 目 : 


$row = mysql_fetch array (S$r); 
print '<form action="delete entry.php" method="post"> 
<p>Are you sure you want to delete this entry?</p> 














D><h3 x Srow Ee" titlerl ms /3 

$row['entry'] . '<br/> 

<input type="hidden" name="id" value="' . $_GET['id'] . '" /> 

<input type="submit" name="submit" value="Delete this Entry!" /></p> 
</form>'; 








和 前 面 例子 中 检索 所 有 记录 并 通过 while 循 环 进行 处 理 不 同 ， 这 里 只 调用 了 一 次 mysal_ 
fetch_array() 畏 数 ， 将 返回 的 记录 赋 给 Srow 变 量 。 使 用 该 数组 可 以 显示 将 要 被 删除 的 记录 。 

该 表单 首先 显示 博客 条 目 , 这 和 view_blog .php 脚 本 所 做 的 非常 类 似 。 当 用 户 单 击 了 按钮 ， 
表单 会 被 提交 回 该 页 面 ， 此 时 记录 将 被 删除 。 完 成 这 一 功能 ， 通 过 $_GET['id'] 传 递 到 脚本 中 
的 blog 条 目的 ia 值 , 必须 存放 到 一 个 隐藏 输入 框 中 , 这 样 在 提交 时 它 会 存在 于 $_POST 数 组 中 ( 因 
为 此 时 $_cET['ig'] 中 不 再 有 值 )。 

(5) 如 果 查 询 失 败 ， 则 报告 错误 : 


} else { // 无 法 获取 信息 。 
print '<p style="color: red;"> Could not retrieve the blog entry because:<br/>' . 
mysql_error($dbc) . '.</p> <p>The query being run was: ' . Squery . '</p>'; 


} 
如 果 SELECT 查 询 运行 失败 ， 则 MySQL 错 误 和 查询 本 身 会 被 打印 出 来 。 
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(6) 检查 表单 是 否 正在 提交 : 

} elseif (isset($ POST['id']) && is _ numeric($_POST['id'])) { // 处 理 表 单 。 

这 个 elseif 子 句 是 第 3 步 中 开始 创建 的 条 件 语 句 的 一 部 分 。 它 对 应 着 该 脚本 的 第 二 个 功能 ( 表 
单 正在 提交 )。 如 果 该 条 件 是 TRUE， 则 应 该 删除 记录 。 

(7) 定义 和 执行 查询 : 


$query = "DELETE FROM entries WHERE entry_id={$_ POST['id']} LIMIT 1"; 
$r = mysql_query ($query, $dbc); 


该 查询 删除 sntry_ia 与 sS_PosTI'ia'] 值 相等 的 记录 。 
这 里 使 用 的 ID 值 来 自 于 表单 ， 存 放 在 其 中 的 隐藏 输入 框 中 。 通 过 向 查询 添加 LIMIT 1 子 句 ， 
可 以 确保 最 多 只 删除 一 条 记录 。 














(8) 检查 查询 的 结果 : 
if (mysql_ affected rows($dbc) == 1) { 
print '<p>The blog entry has been deleted.</p>'; 
} else { 
print '<p style="color: red;"> Could not delete the blog entry because:<br/>' . 
mysSql_error($dbc) . '.</p> <p>The query being run was: ' . Squery . '</p>'; 


} 

mysql_affected_rows() 函数 返回 最 近 一 条 查询 所 修改 的 行 数 。 如 果 查 询 执行 成 功 了 ， 
会 有 一 行 被 删除 ， 所 以 该 函数 应 该 返回 1。 如 果 是 这 样 的 话 ， 则 打印 一 条 消息 。 否 则 ， 出 于 调试 
的 目的 ， 会 打印 出 MySQL 错 误 和 查询 本 身 。 

(9) 完成 主 条 件 语句 : 


) else { // 没有 获取 id。 
print '<p style="color: red;"> This page has been accessed in error.</p>'; 
】 // 结束 主 条 件 语 向 。 


如 果 既 没有 使 用 GET 方 法 也 没有 使 用 POST 方法 将 数值 iq 值 传递 到 页 面 中 ,， 则 这 个 else 会 发 
生 作 用 (参见 图 12-24)。 
(10) 关闭 数据 库 连接 ， 并 完成 页 面 : 


mysql_close(s$dbc); 
?> 

</body> 

</html> 


(11) 将 脚本 保存 为 delete_entry .php， 放 置 在 启用 了 PHP 的 服务 器 上 的 适当 目录 中 ， 并 在 
Web 浏 览 器 中 测试 (参见 图 12-22 和 图 12-23)。 
要 测试 该 脚本 ， 必 须 首 先 运行 view_blog.php。 然 后 单 击 其 中 的 Delete 链 接 访问 delete_ 
entry .php。 -2 
Vv 提示 
口 通过 运行 TRUNCATE TABLE tablename 查 询 可 以 清空 一 个 表 。 这 种 方式 要 优 于 使 用 
DELETE FROM tablename。TRUNCATE 会 完全 删除 并 重建 一 个 表 ， 这 对 数据 库 来 说 更 好 。 
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口 试图 运行 DELETE * FROM tablename 这 样 的 查询 ( 像 编 写 SELECT 查 询 那样 ) 是 一 种 非 
常常 见 的 错误 。 要 记得 ，DELETE 和 SELECT 使 用 的 语法 是 不 同 的 ,因为 你 不 能 删除 指定 的 








列 。 


12.10 更 新 数据 库 中 的 数据 





本 章 将 要 介绍 的 最 后 一 类 查询 是 UPDATE。 它 用 于 修改 一 条 记录 中 某 些 列 上 的 值 。 其 语法 如 


下 所 示 : 














UPDATE tablename SET column1_ name=value, column2 name=value2 WHERE some_ column=value 


和 任何 其 他 查询 一 样 ， 如 果 值 是 字符 串 ， 则 应 该 将 其 放置 在 单 引 号 中 : 














UPDATE users SET first name='Eleanor', 


age=7 WHERE user_id=142 





























和 DELETE 查 询 一 样 ， 应 该 使 用 一 个 wHERE 子 句 来 限制 所 影响 的 行 。 如 果 不 这 么 做 ， 则 数据 


库 中 的 每 条 记录 都 将 被 更 新 。 




















测试 更 新 的 结果 ， 请 再 次 使 用 mysql_affected_rows () 国 数 ， 返 回 修改 的 记录 数量 。 

作为 演示 , 下面 我 们 编写 一 个 页 面 用 于 编辑 博客 条 目 。 该 页 面 允 许 用 户 修 改 条 目的 标题 和 文 
字 ， 但 不 能 修改 输入 日 期 和 博客 idq 值 (作为 主键 ，iq 值 永远 不 应 改变 ) 。 这 个 脚本 使 用 的 结构 和 
delete_entry.php (参见 脚本 12-8) 类 似 ， 首 先 显示 条 目 〈 参 见 图 12-25) ， 然 后 处 理 表单 的 提 


交 (参见 图 12-26)。 








QANN 
) (yy Edit a Blog Entry 
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Edit an Entry 





Entry Title: This is the newest post! (Edited) 





This is so absolutely amazing that rm downright 
speechless! | am still speechless. 


Entry Text: 





( Update this Entry! ) 














@NN 


(Sy Edit a Blog Entry 





4 > C 


© phpvqs4:8888/edit_entry.php 








Edit an Entry 


The blog entry has been updated. 














图 12-25 ” 当 用 户 抵达 编辑 页 时 ， 在 表单 中 显示 
现 有 的 值 











党 更 新 数据 库 中 的 数据 























图 12-26 “继续 提交 表单 ， 用 户 会 看 到 这 样 的 消息 

















(1) 在 文本 编辑 器 或 IDE 中 打开 一 个 新 的 PHP 文 档 ， 命 名 为 edit_entry.php (参看 脚本 








12-9)。 


<!IDOCTYPE html PUBLIC 





"-//W3C//DTD XHTML 1.0 Transitional//EN" 
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"http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
<html xmlns="http://www.w3.0rg/1999/xhtml" xml:lang="en" lang="en"> 
<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<title>Edit a Blog Entry</title> 
</head> 
<body> 
<hl>Edit an Entry</hil> 


脚本 12-9 ”使 用 UPDATE SQL 命令 可 以 编辑 数据 库 表 中 的 记录 






































工 <!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

号 <html xmlns="http://www.w3.0org/1999/xhtml" xml:lang="en" lang="en"> 

4 <head> 

5 <meta http-equiv="content-type" content="text/html; charset=utf-8" /> 

6 <title>Edit a Blog Entry</title> 

7 </head> 

8 <body> 

9 <h1l>Edit an Entry</h1l> 

10 <?php // 脚本 12-9 - edit_entry .php 

11 /* 脚本 使 用 UPDATE 编 辑 博客 条 目 。*/ 

2 

13 // 连接 服务 器 并 选中 数据 库 : 

14 Ss$dbc = mysql_ connect('localhost', 'username', 'password'); 

15 mysql_select db('myblog', $dbc); 

16 

17 if (isset($_GET['id']) && is numeric($_GET['id']) ) { // 在 表单 中 显示 条 目 : 

18 

19 // 定义 查询 。 

20 $query = "SELECT title, entry FROM entries WHERE entry_id={$_GET['id']}"; 

2 if (Sr = mysql_query($Squery，S$dbc)) { // 运行 查询 。 

22 

D3 Srow = mysql_fetch array($r); // 返回 信息 。 

24 

25 // 创建 表单 : 

26 print '<form action="edit_ entry.php" method="post"> 

27 <p>Entry Title: <input type="text" name="title" size="40" maxsize="100" 
value="' . htmlentities($row['title']) . '" /></p> 

28 <p>Entry Text: <textarea name="entry" cols="40" rows="5">' . htmlentities 
(Srow['entry']) . '</textarea></p> 

29 <input type="hidden" name="id" value="' . $_GET['id'] . '" /> 

30 <input type="submit" name="submit" value="Update this Entry!" /> 

en </form>'; 

32 

33 】 else { // 无 法 获取 信息 。 

34 print '<p style="color: red;">Could not retrieve the blog entry because: 

<br/>' . mysql_ error($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 

35 } 

36 

37 } elseif (isset($ POST['id']) && is_ numeric($_POST['id'])) { // 处 理 表 单 。 

38 


39 // 验证 表单 数据 并 确保 安全 : 
40 $sproblem = FALSE; 
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41 if (!empty($_POSTI['title']) && !empty($_POST['entry'])) { 

42 $title =mysql_real escape string(trim(strip tags($_POST['title'])), $dbc); 

43 $sentry =mysql_real _ escape string(trim(strip tags($_POST['entry'])), $dbc); 

44 } else { 

45 print '<p style="color: red;">Please submit both a title and an entry.</p>'; 

46 sproblem = TRUE; 

47 } 

48 

49 IE (!S$Sproblem) { 

50 

51 // 定义 查询 。 

52 $query = "UPDATE entries SET title='$title', entry='$entry' WHERE entry_ id= 
{$_POST['id']}"; 

53 Sr = mysql query ($query，$dbc); // 执行 查询 。 

54 

55 // 打印 结果 : 

56 if (mysql affected rows($dbc) == 1) { 

57 print '<p>The blog entry has been updated.</p>'; 

58 } else { 

59 print '<p style="color: red;">Could not update the entry because:<br/>' . 

mysql_ error($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 

60 } 

61 

62 }】 // 一 切 正常 ! 

63 

64 } else { // 没有 获取 id。 

65 print '<p style="color: red;">This page has been accessed in error.</p>'; 

66 } // End of main IF. 

67 

68 mysql_close($dbc); // 关闭 连接 。 

69 

70 ?> 


71 </body> 
72 </html> 


(2) 打开 PHP 代 码 节 并 连接 到 数据 库 : 


<?php // 脚本 12-9 - edit_entry .php 
$sdbc = mysql_connect('localhost', 'username', 'password'); 
mysql_select db('myblog', $dbc); 


(3) 如 果 页 面 从 URL 中 接收 到 了 有 效 的 条 目 ida， 则 定义 并 执行 一 个 SELECT 查询 : 


if (isset($_GET['id']) && is numeric($_GET['id']) ) { 
$query = "SELECT title, entry FROM entries WHERE entry_id={$_GET['id']}"; 
if ($r = mysql_query ($query, $dbc)) { 


这 段 代 码 和 删除 页 中 的 代码 完全 一 样 ， 针 对 提供 的 ia 值 从 数据 库 中 检索 两 列 值 。 
(4) 检索 记录 ， 在 一 个 表单 中 显示 条 目 : 


Srow = mysdl_fetch_array(Sr) : 
print '<form action="edqit_entry.php" method="post"> 

<p>Entry Title: <input type="text" name="title" size="40" maxsize="100" value="' 
"htmlentities($row['title']) . '" /></p> 

<p>Entry Text: <textarea name="entry" cols="40" rows="5">' 
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htmlentities($row['entry']) . '</textarea></p> 

<input type="hidden" name="id" value="' . S$ GET['id'] . '" /> 
<input type="submit" name="submit" value="Update this Entry!" /> 
</form>'; 


这 段 代 码 也 是 和 前 面 的 脚本 完全 一 样 , 包括 最 重要 的 、 在 表单 的 隐藏 输入 框 中 存放 ia 值 这 一 
步 。 然 而 ,这 里 并 不 是 仅仅 打印 出 存储 的 数据 ， 而 是 将 其 用 作 表 单元 素 的 值 。 为 了 安全 和 避免 潜 
在 的 冲突 ， 每 个 值 在 使 用 之 前 都 用 htmlentities () 先 进行 了 处 理 。 

(5) 如 果 查 询 失败 则 报告 一 个 错误 : 


} else { // 无 法 获取 信息 。 
print '<p style="color: red;">Could not retrieve the blog entry because: 








<br/>' . mysql_ error ($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 
} 
(6) 检查 表单 的 提交 : 
} elseif (isset($_POST['id']) && is numeric($_ POST['id'])) { 





当 表 单 正在 提交 时 ， 该 条 件 将 为 TRUE。 
(7) 验证 表单 数据 并 确保 安全 : 


$sproblem = FALSE; 


if (!empty($_POST['title']) && !empty($_POST['entry'])) { 
$title = mysql_real escape string(trim(strip_ tags($_POST ['title'])), $dbc); 
Sentry = mysql_real _ escape string(trim(strip_ tags($_POST ['entry'])), $dbc); 
} else { 


print '<p style="color: red;"> Please submit both a title and an entry.</p>'; 
$sproblem = TRUE; 
. 


这 段 代 码 来 自 于 用 于 添加 博客 条 目的 页 面 。 它 对 提交 的 数据 进行 了 最 小 的 验证 , 然后 在 其 上 
运行 了 mysql_real_escape_string () 函数 使 其 变 得 安全 。 因 为 表单 数据 是 可 以 编辑 的 ， 所 以 
应 该 像 对 待 创建 新 记录 那样 对 其 进行 验证 。 

(8) 定义 和 执行 查询 : 


IE (!$Sproblem) { 
$query = "UPDATE entries SET title='$title', entry='s$entry’' 
"WHERE entry_id={$_POST['id']}"; 
Sr = mysql query ($query, S$dbc); 


UPDATE 查 询 将 title 列 的 值 设置 为 在 表单 中 名 为 title 的 输入 框 控件 里 输入 的 值 ， 并 将 
entry 列 的 值 设置 为 在 表单 中 名 为 entry 的 textarea 控 件 中 输入 的 值 。 只 有 当 记 录 的 entry_iqd 
等 于 $_POSTI['id']， 这 个 来 自 于 表单 中 的 隐藏 输入 框 时 ， 它 才 会 被 更 新 。 

(9) 报告 查询 的 成 功 执 行 : 


























if (mysql_affected rows($dbc) == 1) { 
print '<p>The blog entry has been updated.</p>'; 
} else { 
print '<p style="color: red;"> Could not update the entry because:<br/>' . mysql_ 


error($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 
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如 果 恰 好 有 一 行 受到 影响 ， 则 会 条 成 功 的 消息 。 否 则 ，MySQL 错 误 和 查询 自身 会 被 
发 送 到 Web 浏 览 器 。 
(10) 完成 条 件 语句 : 
} // 一 切 正常 ! 
) else { // 没有 获取 id。 


print '<p style="color: red;"> This page has been accessed in error.</p>'; 
} // End of main IF. 


如 果 既 没有 通过 GET 方 法 也 没有 通过 POST 方法 将 数字 iq 值 传递 给 该 页 面 , 则 这 个 else 子 句 
会 发 生 作用 。 
(11) 关闭 数据 库 连 接 ， 并 完成 页 面 : 


mysql_close(s$sdbc); 
学 和 

</body> 

</html> 


(12) 将 文件 保存 为 edit_entry .php, 放置 在 启用 了 PHP 的 服务 器 上 的 适当 目录 中 ,并 在 Web 
浏览 器 中 进行 测试 (参见 图 12-25 和 图 12-26)。 
和 前 面 的 示例 一 样 , 要 编辑 一 个 条 目 ， 需 要 首先 访问 view_blog.php 页 面 并 单 击 其 中 的 Edit 
链接 。 
(13) 再 次 访问 view_blog.php， 确 认 已 经 发 生 了 更 改 (参见 图 12-27) 。 
ANMOA 
































(Wy View My Blog 





€ >》 © © phpvgs4:8888/view entries.php 





My Blog 
This is the newest post! (Edited) 


This is so absolutely amazing that Tm downright speechless! I am still speechless. 
Edit Delete 














图 12-27 重新 加 载 view_blog .php 脚 本 ， 查 看 对 条 目 作 出 的 修改 


V 提示 

口 id 是 一 个 主键 ， 这 意味 着 它 的 值 永远 不 应 改变 。 通 过 在 表 中 使 用 主键 ， 可 以 修改 某 一 行 

记录 中 所 有 其 他 列 上 的 值 ， 但 仍然 能 够 用 主键 列 来 引用 这 一 行 记 录 。 

口 无 须 在 了 D 值 上 应 用 mysql_real_escape_string() 国 数 , 因为 is_numeric() 训 试 已 经 

确保 了 它 不 可 能 包含 撤 号 或 其 他 有 问题 的 字符 。 

口 编辑 和 删除 页 中 多 次 在 条 件 语 句 中 使 用 了 mysql_num_rows (), 在 访问 SELECT 查询 的 结 
果 之 前 确保 其 中 有 一 行 数据 : 
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if (mysql_ num rows($r) = = 1) {... 


口 如 果 在 表 中 执行 了 一 个 更 新 ， 但 没有 修改 任何 记录 的 值 ， 则 mysql_affected_rows () 
将 会 返回 0。 
口 不 要 忘记 向 UPDATE 查 询 添加 一 个 LIMIT 1 子 句 ， 确 保 最 多 只 影响 到 一 行 。 


12.11 回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
com/forum/。 





























12.11.1 回顾 


口 你 使 用 的 是 哪个 版 本 的 MySQL? 你 使 用 的 连接 MySQL 的 语句 是 什么 ? 

口 PHP 脚 本 是 如 何 与 MySQL 服 务 器 连接 的 ?如 何 断 开 连 接 ? 

口 什么 是 错误 抑制 运算 符 ， 作 用 是 什么 ? 

口 哪个 函数 可 以 返回 MySQL 报 告 的 错误 ? 

口 在 使 用 PHP 脚 本 与 MySQL 交 互 时 ， 如 果 出 现 问 题 ， 应 该 使 用 什么 调试 技术 ? 

口 如 何 选择 数据 库 ? 

口 创建 一 张 表 的 SQL 命令 是 什么 ?添加 新 记录 、 检 索 记 录 、 修 改 记 录 、 删 除 记 录 的 命令 都 是 
什么 ? 

口 字符 串 值 应 该 使 用 什么 国 数 来 防止 SQL 注入 攻击 ? 








12.11.2 ”实践 


口 阅读 与 你 的 MySQL 版 本 相同 的 MySQL 用户 手 册 。 

口 将 连接 和 选择 数据 库 的 代码 保存 到 一 个 单独 的 脚本 中 , 然后 在 与 数据 库 交 互 的 PHP 页 面 中 
引入 这 个 脚本 。 

口 将 adqdq_entry.php 表 单 改 为 粘性 表单 。 

D 修改 view_entry .php 中 的 代码 ， 将 输入 的 所 有 换行 符 转换 成 HTML 的 换行 标签 。 














将 所 有 的 组 合 在 一 起 








本 章 内 容 

口 准备 开始 

口 连接 数据 库 
口 编写 用 户 自 定义 函数 
口 创建 模板 

口 登录 
口 登 出 

口 添加 名 人 名 言 
口 显示 名 人 名 言 
口 编辑 名 人 名 言 
口 删除 名 人 人 名言 
口 创建 主页 

口 回顾 和 实践 





前 儿童 介绍 了 PHP 用 于 Web 开 发 的 所 有 基础 知识 。 这 一 章 ， 我 们 会 运用 所 有 学 过 的 知识 创建 
一 个 功能 完备 的 网 站 。 当 然 ， 这 一 章 也 会 介绍 一 些 新 的 知识 ,你 会 从 中 掌握 很 多 新 的 技巧 。 特 别 
是 你 会 看 到 如 何 从 零 开 始 开发 一 个 完整 的 Web 应 用 程序 。 


13.1 准备 开始 


在 开始 任何 网 站 项 目 之 前 ,都 要 先 制定 目标 。 这 一 章 的 主要 目标 是 应 用 前 面 学 过 的 所 有 知识 
(绝对 是 一 个 崇高 的 目标 )。 这 一 章 的 例子 会 将 前 两 章 使 用 的 模块 组 合 在 一 起 。 我们 会 创建 一 个 网 
站 ， 该 网 站 可 以 存储 和 显示 名 人 名 言 。 本 章 不 会 像 第 11 章 那样 使 用 文件 存储 名 人 名 言 ， 而 是 用 
MySQL 数 据 库存 储 用 户 提交 的 名 人 名 言 。 就 像 第 12 章 的 博客 示例 一 样 ， 这 一 章 也 会 实现 创建 、 
编辑 和 删除 名 人 名 言 。 默 认 情 况 下 ， 公 共用 户 可 以 在 主页 上 看 到 最 新 添加 的 名 言 (参见 图 13-1)， 
主页 上 也 可 以 显示 随机 名 言 或 随机 的 受 欢迎 的 名 言 。 

为 了 提高 网 站 安全 性 ， 这 个 网 站 会 有 一 个 管理 员 ， 可 以 登录 和 登 出 网 站 。 只 有 以 管理 员 身 份 























13.1 0000 345 





登录 后 ， 才 可 以 创建 、 编 辑 和 删除 名 言 (参见 图 13-2)。 


2 My Site of Quotes - Mozilla Firefox 


My Site of Quotes 


3) Edit a Quote - Mozilla Firefox 


My Site of Quotes 
Access Denied! 


You do not have permission to access this page 














图 13-1 网 站 简单 的 主页 图 13-2” 非 管理 员 会 被 拒绝 访问 某 些 页 面 


这 个 网 站 会 使 用 一 个 简单 的 模板 ， 以 保证 所 有 页 面 的 显示 效果 一 致 。 我 们 会 使 用 CSS 处 理 所 
有 版 式 和 布局 。 网 站 还 会 使 用 一 个 用 户 定义 函数 ， 我 们 将 这 个 函数 写 在 一 个 包含 文件 中 。 
和 第 12 章 一 样 ， 我 们 会 首先 创建 数据 库 和 一 个 表 。 数 据 库 可 以 命名 为 myquotes (你 也 可 以 自 
己 起 个 名 字 )。 用 以 下 SQL 命 令 创 建 数据 表 : 
CREATE TABLE quotes ( 
quote_id INT UNSIGNED NOT NULL AUTO_INCREMENT ， 
Guote TEXT NOT NULL, 
source VARCHAR(100) NOT NULL, 
favorite TINYINT(1) UNSIGNED NOT NULL, 
date_entered TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, 
PRIMARY KEY (quote_id) 
) 


quote_id 是 主键 ,每 当 添 加 一 个 新 的 名 言 时 ， 这 个 值 都 会 递增 。quote 字 段 用 于 存储 名 言 ， 
source 字 段 用 于 存储 名 言 出 处 ， 即 人 名 (不同 于 第 11 音 将 名 言 和 人 名 一 起 存储 )。favorite 字 
段 存储 1 或 0， 用 于 标记 名 言 是 否 受 欢迎 。dqate_entereq 字 段 是 一 个 时 间 惟 ， 当 创建 一 个 新 记录 
时 会 自动 设置 当前 时 间 戳 。 

可 以 使 用 PHP 脚 本 创建 这 个 表 (参看 脚本 12-4), 也 可 以 使 用 第 三 方 应 用 程序 (例如 , MySQL 
客户 端 或 phpMyAdmin ) 。 

最 后 介绍 一 下 服务 器 中 网 站 的 文件 组 织 结构 (参见 图 13-3)。 具 体 地 说 ， 用 于 创建 数据 库 连 
接 的 mysql_connect .php 脚 本 应 该 存储 在 Web 根 目录 的 外 部 ， 如 果 条 件 不 允许 这 样 做 ， 可 以 把 
它 放 在 includes 文 件 夹 下 面 ， 注 意 修改 其 他 相关 脚本 中 的 代码 。 

创建 完 数 据 库 、 数 据 库 表 和 必要 的 文件 夹 之 后 ， 就 可 以 开始 编码 工作 了 。 
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Web 根 目录 
-一 一 一 一 三 
add quote.php 
css 
| 上 一 的 sylecss 
delete_quote.php 


mysql_connect.php Ww edit_quote.php 


includes 

ee functions.php 
index.php 
login.php 
logout.php 

templates 仿 footerhtml 

[ 广 要 ee 

上 = > 做 headerhtml 


view_quotes.php 


13-3 ”服务 器 上 的 文件 组 织 结 构 


13.2 ”连接 数据 库 


在 第 12 章 中 ,为 了 连接 数据 库 ， 我 们 在 所 有 脚本 中 添加 了 同样 的 代码 。 这 回 我 们 使 用 更 常用 
的 方法 ， 就 是 将 那些 需要 重复 使 用 的 代码 放 到 一 个 独立 的 文件 中 。 每 个 与 数据 库 交 互 的 脚本 〈 大 
部 分 脚本 ) 都 会 包含 这 个 文件 。 如 图 13-3 所 示 ， 数 据 库 连 接 文件 应 当 放 在 Web 根 目录 的 外 面 ， 如 
果 条 件 不 允许 你 这 样 做 ， 你 可 以 把 它 放 在 includes 文 件 夹 下 面 。 


过 创建 mysql_connect .php 














(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 脚本 ,命名 为 mysql_connect .php (参看 脚本 13-1): 


<?php 





133 DUDUUUDO 347 





脚本 13-1 这 个 脚本 连接 数据 库 服 务 器 ， 并 选择 使 用 的 数据 库 


1 <?php // 脚本 13-1 - mysdql_connect .php 

2  /* 脚本 连接 服务 器 并 选中 数据 库 。*/ 

3 

4  // 连接 : 

局 $sdbc = mysql_connect('localhost', 'username', 'password'); 
6 

7 // 选中 : 

8 mysql_select_ db('myquotes', $dbc); 

9 

下 0 3 

(2) 连接 数据 库 服务 

$dbc = mysql_ connect('localhost', 'username', 'password'); 


你 可 以 根据 自己 使 用 的 服务 器 情况 改变 这 些 值 。 
(3) 选择 数据 库 : 

mysql_select db('myquotes', S$dbc); 

你 可 以 根据 要 连接 的 数据 库 名 称 ， 修 改 相 应 的 值 。 
(4) 结束 脚本 : 


> 


(5) 将 脚本 保存 为 nysql_connect .php。 


13.3 ”编写 用 户 定 义 函 数 


这 个 网 站 包含 一 个 用 户 定义 函数 。 如 第 10 章 所 述 ， 如 果 脚 本 或 网 站 中 有 很 多 重复 的 代码 ， 最 
好 创建 自 定 义 函数 。 这 个 网 站 有 很 多 这 种 情况 ,最 明显 的 例子 是 : 很 多 脚本 都 需要 检查 当前 用 户 
是 否 为 管理 员 。 下 面 的 脚本 会 创建 用 户 定义 函数 ， 它 返回 一 个 布尔 值 ， 表 示 用 户 是 否 为 管理 员 。 
但 是 脚本 如 何 测试 是 不 是 管理 员 呢 ? 

如 果 成 功 登 录 网 站 ， 服 务 器 会 将 一 个 名 为 es 
ee ie 13-4) 全 wo site 10. . 1 rp ee 人 a cookie. 

管理 员 的 浏览 器 。 这 可 能 看 起 来 很 奇怪 或 随机 , 但 i mr Tr 
实 就 是 如 此 。 当 你 使 用 简单 的 方式 〈 比 如 cookie) 做 和 
验证 时 , 最 好 使 用 这 种 不 太 明 显 的 验证 组 合 。 如 果 用 外 

很 清楚 的 验证 组 合 ， 比 如 名 为 admin 值 为 TRUE, 那 | se 全 ea 
么 任何 人 都 可 以 狂 到 它 并 伪造 它 。 了 解 了 这 些 之 后 ， ed ah 
以 下 的 函数 只 会 检查 是 否 存在 名 为 Samuel、 值 为 图 13-4 ”这 个 cookie 用 于 辨认 管理 员 


Clemens 的 cookie 。 











Name: Samuel 
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(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 脚本 ， 命 名 为 functions .Php (参看 脚本 13-2) : 
<?php // 脚本 13-2 - functions .php 
虽然 这 个 脚本 只 定义 了 一 个 自 定义 函数 ， 但 却 用 了 复数 形式 (functions) 命名 ， 因 为 将 来 可 
能 需要 向 这 个 脚本 中 添加 其 他 用 户 自 定义 函数 。 
脚本 13-2 is_agdministrator() 函数 定义 在 了 一 个 可 包含 脚本 中 ， 每 个 需要 验证 管理 员 状 态 
的 页 面 都 可 以 调用 此 脚本 


<?php // 脚本 13-2 - functions .php 
/* 脚本 定义 自 定 义 函 数 。*/ 











// 函数 检查 用 户 是 否 为 管理 员 
// 也 数 接受 两 个 可 选 参数 。 
// 函数 返回 一 个 布尔 值 。 


function is_adqdministrator(Sname = 'Samuel'，S$value = 'Clemens') { 


\D oo、~OUwU 必 wmN 


// 检查 cookie 是 否 存 在 和 cookie 值 : 
if (isset(S$_COOKIE [Sname]) && ($_COOKIE [Sname] == S$value)) { 
return 七 YUe 
} else { 
return false; 








} 


} // 结束 1s_administrator () 有 函数 。 


auwm 必 wb Po 





OO 


(2) 定义 一 个 新 函数 

function is administrator($name = 'Samuel', S$Svalue = 'Clemens') { 

国 数 接受 两 个 参数 : cookie 的 名 和 值 。 两 个 参数 都 有 默认 值 。 

由 于 这 个 函数 只 检查 某 个 特定 值 的 cookie， 所 以 其 实 并 没有 必要 设置 参数 及 其 默认 值 。 设 置 
这 些 参数 是 出 于 将 来 扩展 网 站 功能 的 目的 〈 比 如 ， 如 果 你 要 执行 多 种 类 型 的 验证 )。 使 用 带 默认 
值 的 参数 ， 可 以 在 调用 函数 时 省 去 参数 值 ， 前 提 是 已 经 验证 过 它们 。 

(3) 根据 cookie 是 否 存 在 和 cookie 的 值 ， 返 回 布 尔 值 : 


























if (isset(S$_COOKIE [Sname]) && ($_ COOKIE[S$name] == $value)) { 
return true; 
} else { 


return false; 


} 
如 果 cookie 存 在 ， 并 且 cookie 值 正确 ， 会 返回 TRUE， 耕 则 函数 返回 FALSE。 
(4) 结束 函数 和 脚本 : 


} // 结束 is_administzratozr () 函 数 。 


?> 
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(5) 将 脚本 保存 为 functions .php， 存 放 在 includqes 目 录 下 。 


13.4 创建 模板 
我 们 已 经 创建 了 两 个 辅助 文件 ， 接 下 来 创建 模板 。 如 第 8 章 所 述 ， 网 站 的 布局 由 两 个 可 包含 


文件 控制 : 头 文 








件 和 页 脚 文 件 。 这 两 个 文件 会 存放 在 


templates 目 录 中 。 头 文件 会 引用 一 个 样式 表 ， 样 式 表 
文件 存放 在 css 文 件 夹 下 。 你 可 以 在 本 书 第 13 章 的 源 代 码 
(ch13 文 件 夹 下 ) 中 找到 这 个 样式 表 。( 源 代码 可 以 从 
www.LarryUllman.com 下 载 。) 

除了 产生 网 页 主要 的 HTML 之 外 ， 头 文件 还 包含 自 
定义 函数 脚本 。 如 果 当 前 用 户 是 管理 员 ， 页 脚 文件 还 会 
显示 一 些 管理 链接 (参见 图 13-5)。 


这 创建 头 文件 



































图 13-5 如 果 访 问 页 面 的 用 户 是 管理 员 ， 


他 会 看 到 一 些 管理 链接 








(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 HTML 文 档 ， 命 名 为 header .html (参看 脚本 13-3): 


<?php // 脚本 13-3 - header.html 
include('includes/functions.php'); ?> 


头 文件 以 PHP 代 码 开头 ， 目 的 是 包含 functions .php 肢 本， 网 站 中 的 很 多 页 面 都 会 用 到 此 


脚本 。 使 用 相对 路 径 引用 被 包含 脚本 ， 因 





脚本 13-3 ” 头 文 件 包含 functions .php 脚 本 ， 并 创建 HTML 页 面 


<head> 


OOOOONRODOPp 


} 


OIAWVOPO 


</head> 
<body> 





\O 


<meta http-equiv="content-type" 
<link rel="stylesheet" media="all" href="css/style.css" /> 
<title><?php // 打印 页 面 标 题 。 

if (defined('TITLE')) { // Is the 上 title 


<?php // 脚本 13-3 - header.html 


// 包含 functions 脚 本 : 
include('includes/functions.php'); ?> 
<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.0org/1999/xhtml" 





print TITLE; 


} else { // 标题 未 定义 。 
print 'My Site of Quotes'; 


?></title> 


20 <div id="container"> 
2 <hl>My Site of Quotes</h1i> 
有 22 <br/> 


content="text/html; 


xml :lang="en" 


defined? 


为 包含 头 文 件 的 页 面 都 存放 在 主 目 录 中 。 





lang="en"> 


charset=utf-8" /> 
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23 <!-- 可 变 内 容 开始 。--> 
(2) 开始 编写 HTML 文 档 : 


<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtml1/ DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.0org/ .1999/xhtml" xml:lang="en"lang="en"> 

<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<link rel="stylesheet" media="all" href="css/style.css" /> 


可 以 从 本 书 的 源 代码 中 找到 这 个 样式 表 。 使 用 相对 路 径 引 用 样式 表 ， 因为 包含 头 文 件 的 页 面 
都 存放 在 主 目录 中 。 
G) 打印 页 面 标题 


<title><?php 
if (defined('TITLE')) { 
print TITLE; 
} else { 
print 'My Site of Quotes'; 
} 


?></title> 


这 段 代 码 和 第 8 章 类 似 ， 根 据 是 否定 义 了 title 常 量 ， 打 印 默 认 标 题 或 自 定义 标题 
(4) 结束 head 标 签 : 


</head> 


(5) 开始 页 面 body 部 分 : 


<body> 
<div id="container"> 
<hl>My Site of Quotes</h1i> 
<br/> 
<!-—- BEGIN CHANGEABLE CONTENT. --> 


(6) 将 脚本 保存 为 header .html， 存 放 在 templates 目 录 下 。 


过 创建 footer .html 
(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 HTML 文 档 ， 命 名 为 footer .html (参看 脚本 13-4): 
<!-- 可 变 内 容 结束 。 
























































脚本 13-4 ”页 脚 文件 在 恰当 的 时 候 显示 常用 管理 链接 ， 同 时 完成 HTML 页面 


<!-- END CHANGEABLE CONTENT. --> 
<?php // 脚本 13-4 - footer.html 


// 显示 管理 链接 …… 

// - 如 果 用 户 是 管理 员 且 当前 页 面 不 为 1ogout.php 

// - 或 者 $1oggedin 变 量 为 TRUE (例如 ， 用 户 刚刚 登录 ) 

if ( (is administrator() && (basename($_SERVERI['PHP_ SELF']) != 'logout.php')) 
OR (isset($loggedin) && Sloggedin) ) { 


\D oOUORODP 
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10 // 创建 链接 : 

下 print '<hr /><h3>Site Admin</h3><p><a href="add quote.php">Add Quote</a> <-> 
12 <a href="view quotes.php">View All Quotes</a> <-> 
13 <a href="logout.php">Logout</a></p>'; 

14 

L157 .} 

16 

和 ”这 洋 

18 </div><!-- 容器 --> 

19 <div id="footer">Content &copy; 2011</div> 

20 </body> 


21 </html> 
(2) 检查 是 否 需 要 显示 常用 管理 链接 : 

















<?php 
If ( (is administrator() && (basename($_SERVER['PHP_SELF'])!= 'logout.php')) 
OR (isset($loggedin) && S$loggedin) ) { 








这 个 条 件 语 句 有 些 复杂 。 虽 然 很 多 页 面 都 可 以 通过 调用 is_agdministrator () 函数 确认 当 
前 用 户 是 否 为 管理 员 ， 但 是 由 于 cookie 的 工作 方式 ， 在 以 下 两 个 页 面 中 这 个 函数 会 返回 不 正确 的 
结果 : login.php 和 logout .php。 在 logout .php 页 面 上 ， 脚 本 会 在 收 到 管理 员 cookie 之 后 删 
除 它 。 也 就 是 说 ， 管 理 员 在 该 页 面 中 会 看 不 到 管理 链接 (因为 当 他 们 访问 这 个 页 面 时 cookie 已 经 
不 存在 了 )。 条 件 语句 的 前 面部 分 需要 is_administrator() 函数 返回 TRUE 并 且 当 前 页 不 能 是 
logout .php。basename ($_SERVER['PHP_SELF'] ) 代码 是 获取 当前 脚本 的 可 靠 方法 。( 由 于 
页 脚 文 件 包 含 在 男 一 个 脚本 中 ，$_SERVER[' PHP_SELF'] 也 有 口 脚本 的 值 ,) 

条 件 语句 的 后 半 部 分 (OR 之 后 )， 检 查 是 否 设 置 了 $loggedin 晶 其 值 是 否 为 TRUE。 这 是 为 
了 应 对 当 用 户 成 功 登 录 1o0gin .php 页 面 时 的 cookie 问 题 。 is_agdministrator () 函数 在 用 户 刚 刚 
登录 后 不 会 返回 TRUE， 因 为 这 时 cookie 刚 刚 通 过 脚本 发 送 到 客户 端 ， 所 以 不 可 能 被 读 取 到 。 

(3) 创建 链接 : 


print '<hr /><h3>Site Admin</h3><p><a href="add quote.php">Add 
一 QuUote</a> <-> 

<a href="view quotes.php">View All Quotes</a> <-> 

<a href="logout.php">Logout</a></p>'; 


创建 了 3 个 链接 ，Add Quote (用 于 添加 新 名 言 )、View All Quotes (查看 所 有 名 言 ) 和 Logout 
( 登 出 )。 

(4) 完成 条 件 语句 和 PHP 代 码 : 

} 


























2 

(5) 完成 HTML 页 面 : 
</div><!-- 容器 --> 
<div id="footer">Content &copy;2011</div> 
</body> 





</html> 
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(6) 将 文件 保存 为 footer .html， 并 存放 在 templates 目 东 下 。 


13.5 登录 


接 下 来 ， 创 建 管理 员 登 录 脚 本 。 最 终结 果 会 和 第 9 章 的 脚本 类 似 ， 只 是 有 一 个 结构 上 的 区 别 ; 
第 9 章 使 用 的 是 0 DDD ， 人 许 任意 安排 脚本 。 本 章 的 示例 不 使 用 输出 缓存 ， 因 此 在 处 理 表单 时 ， 
必须 保证 发 送 cookie 而 不 导致 headers already sent erros 错 误 〈 参 见 第 9 章 ) 。 换 名 话说， 脚本 在 包含 
头 文件 之 前 会 先 检查 表单 提交 情况 。 为 了 仍然 能 够 捕捉 到 页 面 中 的 错误 信息 和 其 他 信息 ,你 必须 
使 用 几 个 变量 。 


这 创建 头 文 件 




















(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 PHP 文 档 ， 命 名 为 login .php (参看 脚本 13-5 ) : 
<?php // 脚本 13-5 - login.php 
脚本 13-5 ”登录 页 面 会 显示 和 处 理 表 单 ， 在 成 功 登 录 后 会 发 送 cookie 


<?php // 脚本 13-5 - login.php 
/* 页 面 用 于 用 户 登 录 网 站 。*/ 


1 

2 

3 

4  // 使 用 默认 值 定义 两 个 变量 : 
与 $loggedin = false; 

6 Serror = false; 

学 

8 

9 

{ 


// 检查 表单 是 否 已 提交 : 








IE ($_SERVERI['REQUEST METHOD'] == 'POST') 
0 
二 // 处 理 表单 : 
12 IE (!empty($_POST['email']) && !empty($_POST['password'])) { 
3 
4 if ( (strtolower($_ POST['email']) == 'me@example.com') && ($_POST['password'] 
== 'testpass') ) { // 相等 ! 
5 
16 // 创建 cookie: 
7 setcookie('Samuel', 'Clemens', time()+3600); 
18 
19 // 标识 已 登录 : 
20 $loggedin = true; 
21 
22 } else { // 不 相等 ! 
23 
24 $error = 'The submitted email address and password do not match those 
on filel'; 
2.5 
26 } 
27 
28 ) else { // 表单 未 填 完 整 。 
29 


30 $error = 'Please make sure you enter both an email address and a password!'; 
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3 

32 } 

33 

4 

35 

36 // 设置 页 面 标题 并 包含 页 头 文 件 : 

37 define('TITLE', 'Login'); 

38 include('templates/header.html'); 

39 

40 // 在 出 现 错误 时 打印 错误 信息 : 

41 if ($error) { 

42 print '<p class="error">' . Serror . '</p>'; 
43 } 

44 

45 // 检查 用 户 是 否 已 登录 ， 如 没有 登录 则 显示 表单 : 

46 if ($loggedin) { 

47 

48 print '<p>You are now logged in!</p>'; 

49 

50 } else { 

51 

52 print '<h2>Login Form</h2> 

53 <form action="login.php" method="post"> 

54 <p><label>Email Address <input type="text" name="email" /></label></p> 
55 <p><label>Password <input type="password" name="password" /></label></p> 
56 <p><input type="submit" name="submit" value="Log In!" /></p> 
57 </ form 

58 

5 

60 

61 include('templates/footer.html'); // 包含 页 脚 文件 。 
62 ?> 

(2) 使 用 默认 值 定义 两 个 变量 : 


$loggedin = false; 
$error = false; 


脚本 接 下 来 会 使 用 这 两 个 变量 。 这 里 设置 了 变量 的 默认 值 , 表示 用 户 没有 登录 , 也 没有 发 生 
省 误 。 


(3) 检查 表单 是 否 已 经 提交 : 


IE ($_SERVER['REQUEST METHOD'] =='POST') { 

(4) 处 理 表单 : 

if (!empty($_POST['email']) && !empty($_POST['password'])) { 
If ( (strtolower($_POST['email']) == 'me@example.com') && ($_POST 
+['password'] == 'testpass') ) { 


类 似 第 9 童 的 示例 ， 这 个 脚本 首先 确认 $_POsST[' email'] 和 $_PoSTI'passwora'] 非 空 
第 二 个 条 件 语句 检查 提交 数据 是 否 符合 要 求 。 
(5) 创建 cookie: 


setcookie('Samuel', 'Clemens', time()+3600); 
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这 个 cookie 的 名 为 Samuel， 值 为 Clemens。 设 置 了 1 小 时 的 过 期 时 间 。 
(6) 显示 用 户 已 登录 : 

$loggedin = true; 

这 个 脚本 的 后 面 和 页 脚 文件 中 (参看 脚本 13-4) 都 会 使 用 这 个 变量 。 
(7) 为 其 他 两 个 条 件 分 支 创建 错误 信息 : 


} else { // 不 相等 ! 





$error = 'The submitted email address and password do not match those on file!'; 
} 
} else { // 表单 未 填 完 整 。 
$error = 'Please make sure you enter both an email address and a password!'; 


} 
} 


第 1 个 else 子 句 处 理 提 交 了 错误 的 邮件 地 址 和 密码 的 情况 。 第 2 个 else 子 句 处 理 没有 提交 邮 
件 地 址 或 密码 的 情况 ,这 两 种 情况 都 将 消息 发 送 给 一 个 变量 ,这 个 变量 会 在 脚本 后 面 的 代码 中 用 到 。 

(8) 设置 页 面 标题 并 包含 头 文件 : 

define('TITLE', 'Login'); 

include('templates/header.html'); 


(9) 在 出 现 错误 时 ， 打 印 错 误 信 息 : 


if (SEerror) { 
print '<p class="error">' . S$error . '</p>'; 


} 
省 误 的 类 型 会 在 第 (7) 步 时 确定 ,但 并 没有 在 第 (7) 步 打印 出 来 ， 因 为 那 时 还 没有 包含 头 文 件 。 
解决 方法 是 让 代码 检查 非 FALSE 的 $error 值 , 然后 在 HTML 和 CSS 中 打印 $error (参见 图 13-6)。 




















My Site of Quotes 
The submitted email address and password do not match those on file! 


Login Form 


| 























图 13-6 ”在 包含 头 文件 之 后 ，HTML 表 单 之 前 显示 错误 信息 


(10) 显示 用 户 已 登录 ， 否 则 显示 登录 表单 : 


if ($loggedin) { 

print '<p>You are now logged in!</p>'; 

else { 

print '<h2>Login Form</h2> 

<form action="login.php" method="post"> 

<p><label>Email Address <input type="text" name="email" /> </label></p> 
<p><label>Password <input type="password" name="password" /></label></p> 














-一 
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<p><input type="submit" name="submit" value="Log In!" /></p> 
</form>'; 


} 

如 果 $1oggedin 变 量 的 值 为 TRUE (默认 值 为 FALSE)， 就 表明 用 户 已 经 成 功 登 录 网 站 ， 这 
十 会 显示 登录 成 功 的 信息 (参见 图 13-7)。 如 果 $1oggedin 变 量 的 值 仍 为 FALSE， 就 会 显示 登录 
表单 (参见 图 13-8)。 



































My Site of Quotes My Site of Quotes 
Login Form 
| 
Bdd Quote View All Quotes Logout | 
图 13-7 登录 成 功 后 的 页 面 图 13-8 ”基本 登录 表单 





(11) 包含 页 脚 文件 并 完成 页 面 : 


include('templates/footer.html'); 
?> 


(12) 将 文件 保存 为 1ogin.php。 

(13) 在 Web 浏 览 器 中 测试 文件 (参见 图 13-6、 图 13-7 和 图 13-8)。 

我 故意 没有 创建 连接 到 登录 部 分 的 链接 〈 出 于 安全 方面 的 考虑 ) ， 你 需要 在 Web 浏 览 器 的 地 
址 栏 中 输入 正确 的 URL。 





ww 提示 
口 注意 ， 这 个 脚本 需要 管理 员 在 1 小 时 之 后 再 次 登录 ， 无 论 他 是 否 继续 在 网 站 中 活动 。 
13.6 ” 登 出 


既然 写 了 登录 程序 ， 就 应 该 有 登 出 程序 。 以 下 是 一 个 简单 的 登 出 脚本 ， 只 使 用 了 cookie。 


地 创建 logout .php 








(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 PHP 文 档 ， 命 名 为 logout .php (参看 脚本 13-6): 


<?php // 脚本 13-6 - logout.php 


脚本 13-6 登 出 脚本 删除 管理 员 标 识 cookie 
1 <?php // 脚本 13-6 - logout.php 
2 /* 这 是 登 出 页 面 ， 它 会 删除 cookie。*/ 
3 
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4 // 如 果 cookie 存 在 ， 则 删除 : 
5 if (isset($ COOKIE['Samuel'])) { 
6 setcookie('Samuel', FALSE, time()-300); 
7 } 
8 
9 ” // 定义 页 面 标题 并 包含 页 头 文件 : 
10 define('TITLE', 'Logout'); 
11 include('templates/header.html'); 
12 
13 // 打印 一 条 消息 : 
14 print '<p>You are now logged out.</p>'; 
15 
16 // 包含 页 脚 文件 : 
17 include('templates/footer.html'); 
8 ?> 





(2) 如 果 cookie 存 在 ， 删 除 它 : 


if (isset($_ COOKIE['Samuel'])) { 
setcookie('Samuel', FALSE, time()-300); 
} 





为 了 安全 起 见 ， 这 个 脚本 只 会 在 cookie 已 经 存在 时 才 删 除 它 。 


黑客 可 以 在 不 登录 的 情况 下 访 


pn 








问 登 出 脚本 来 获取 cookie 名 ， 而 以 上 查询 语句 则 可 以 防止 出 现 这 种 安全 问题 。 


要 删除 已 经 存在 的 登录 cookie， 可 以 发 送 男 一 个 同名 cookie， 将 值 设置 为 FALSE， 并 将 过 期 





时 间 设 置 为 一 个 过 去 的 时 间 。 
(3) 定义 页 标题 并 包含 头 文件 : 





define('TITLE', 'Logout'); 
include('templates/header.html'); 

(4) 打印 登 出 信息 : 

print '<p>You are now logged out.</p>'; 
(5) 包含 页 脚 文件 : 
include('templates/footer.html'); 

?> 


(6) 将 文件 保存 为 1ogout .php。 
(7) 在 Web 浏 览 器 中 测试 文件 (参见 图 13-9 和 图 13-10)。 








Confirm setting cookie 


The site 10.0.1.2:8888 wants to modify an existing cookie. 


Ss) [| Use my choice for all cookies from this site 


Hide Details Allow for Session Deny 





Name; Samuel 
Content: deleted 
Host; 10,0,1,2 
Path: jch13/htdocs| 
Send For; Any type of connection 
Expires: Wednesday, January 13, 2010 6:32;12 PM 








My Site 


of Quotes 





图 13-9 ”这 个 cookie 删 除 已 经 存在 的 登录 cookie 





| 











13-10 





登 出 后 的 页 面 
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13.7 添加 名 人 和 名言 


现在 管理 员 已 经 可 以 登录 网 站 了 ， 也 可 以 添加 名 言 了 。 这 个 脚本 与 第 12 章 的 添加 blog 文 章 的 
示例 类 似 ， 只 是 脚本 中 的 表单 会 增加 一 个 复 选 框 ， 用 于 标记 引用 语 是 否 是 受 欢迎 的 名 人 名 言 ( 参 
见 图 13-11)。 

















My Site of Quotes 


Add a Quotation 
IIt is the mark of an educated 
mind to be able to entertain a 
thought without accepting it. 





[aristotle 


回 








图 13-11 添加 名 人 名 言 到 数据 库 的 表单 
由 于 这 个 脚本 要 向 数据 库 中 添加 记录 ， 所 以 要 特别 注意 安全 问题 。 首 先 ， 只 有 管理 员 才 可 以 
使 用 这 个 页 面 。 其 次 ， 需 要 验证 SQL 命令 中 使 用 的 值 ， 防 止 无 效 查 询 。 
> 创建 aaa_quote.ph| 


(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 PHP 文 档 ， 命 名 为 a4d_quote .php (参看 脚本 13-7): 


<?php // 脚本 13-7 -adqd_quote.php 
define('TITLE', 'Add a Quote'); 
include('templates/header.html'); 
print '<h2>Add a Quotation</h2>'; 





























脚本 13-7 adg_quote.php 脚 本 只 允许 管理 员 添 加 新 的 名 人 名 言 到 数据 库 中 





1 <?php // 脚本 13-7 - add_quote.php 
2  /* 脚本 添加 名 人 名 言 。*/ 

3 

4  // 定义 页 面 标题 并 包含 页 头 文件 。 

5 define('TITLE', 'Add a Quote'); 

6 include('templates/header.html'); 
7 

8 print '<h2>Add a Quotation</h2>'; 
9 


10 // 强制 只 有 管理 员 可 以 访问 该 页 面 : 
11 if (!is administrator()) { 
12 print '<h2>Access Denied!</h2><p class="error">You do not have permission 
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to access this page.</p>'; 














13 include('templates/footer.html'); 

14 exit(); 

1 5 

16 

17 // 检查 表单 是 否 提 交 : 

18 if ($_SERVER['REQUEST METHOD'] == 'POST') { // 处 理 表单 。 

于 9 

2:0 If ( !empty($_POST['gquote']) && !empty($_ POST['source']) ) { 

21 

22 // 包含 数据 库 连 接 。 

23 include('../mysqgql_connect.php'); 

24 

25 // 准备 查询 中 使 用 的 值 : 

26 Squote = mysql_ real _ escape_ string(trim(strip_ tags($_ POST['quote'])), $dbc); 

2 $source = mysql_ real _ escape string(trim(strip tags($_POST['source'])), $dbc); 

28 

29 // 创建 favorite 值 : 

30 if (isset($_POST['favorite'])) { 

3 $favorite = 1; 

32 } else { 

33 Sfavorite = 0; 

34 } 

35 

36 Sauery = "INSERT INTO quotes (quote, source, favorite) VALUES ('S$quote', 

'$Ssource', S$favorite)"; 

3 水 Sr = mysql_ query ($query, S$dbc); 

38 

39 If (mysql_ affected rows($dbc) == 1) { 

40 // 打印 一 条 消息 : 

41 print '<p>Your quotation has been stored.</p>'; 

42 } else { 

43 print '<p class="error">Could not store the quote because:<br/>' 
mysql_error ($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 

44 } 

45 

46 // 关闭 连接 : 

47 mysql_close(s$sdbc); 

48 

49 】 else { // 没有 填写 名 人 名 言 。 

50 print '<p class="error">Please enter a quotation and a source!</p>'; 

51 } 

52 

53 } // 结束 提交 条 件 语句。 

54 

55 // 结束 PHP 节 并 显示 表单 : 

56” ?> 

2 

58 <form action="add quote.php" method="post"> 

59 <p><label>Quote <textarea name="quote" rows="5" cols="30"> 

</textarea></label></p> 
60 <p><label>Source <input type="text" name="source" /></label></p> 
61 <p><label>Is this a favorite? <input type="checkbox" name="favorite" value= 


"yes" /></label></p> 
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62 <p><input type="submit" name="submit" value="Add This Quote!" /></p> 
63 </form> 

64 

65 <?php include('templates/header.html'); ?> 

(2) 如 果 用 户 不 是 管理 员 ， 将 不 能 访问 这 个 页 面 : 

if (!is administrator()) { 


print '<h2>Access Denied!</h2> <p class="error">You do not have permission to access 
rthis page.</p>'; 

include('templates/footer.html'); 

exit(); 




















蕊 一 


过 调用 is_administrator() 国 数 ， 脚 本 可 以 检查 用 户 是 否 为 管理 员 。 如 果 用 户 不 是 管 
理 员 , 会 显示 Access Denied 错 误 信 息 , 后 面 的 代码 用 于 包含 页 脚 文件 和 结束 脚本 (参见 图 13-12 ) 。 











My Site of Quotes 


Add a Quotation 


Access Denied! 


You do not have permission to access thls page. 











图 13-12 ”如 果 用 户 不 是 管理 员 ， 将 不 能 访问 这 个 页 面 


(3) 检查 表单 提交 情况 : 





IE ($_SERVER['REQUEST METHOD'] =='POST') { 
(4) 检查 表单 中 的 值 : 
if ( lIempty($_POST['quote']) &&!empty($_ POST['source']) ) { 


这 个 脚本 只 进行 了 最 少量 的 表单 验证 , 检查 两 个 变量 是 否 为 空 。 因 为 要 输入 的 是 一 大 块 文本 
(名 人 名 言 )， 也 没有 其 他 什么 需要 验证 的 内 容 。 





(5) 准备 查询 中 使 用 的 值 : 
$Squote = mysql_real escape string(trim(strip tags($_ POST['guote'])), S$dbc); 
$source = mysql_ real escape_ string(trim(strip tags($_POST['source'])), $dbc); 


为 了 使 这 两 个 文本 值 能 够 在 查询 中 安全 地 使 用 ， 需 要 使 用 mysql_real_escape_string() 
函数 (参见 第 12 章 )。 为 了 使 这 些 值 之 后 可 以 在 Web 浏 览 器 中 正常 显示 , 需要 使 用 strip_tags () 
函数 〈 参 见 第 5 章 ) 。 

(6) 创建 favorite 值 : 


if (isset($_POST['favorite'])) { 
Sfavorite = 1; 
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} else { 
Sfavorite = 0; 


} 
在 数据 库 中 ， 名 人 名 言 的 受 欢迎 状态 用 1 和 0 表示 : 1 为 受 欢迎 ，0 为 不 受 欢 迎 。 为 了 确定 使 用 


哪个 数字 ， 所 有 的 PHP 脚 本 都 会 检查 s_POST ['favorite'] 的 值 。 如 果 设 置 了 这 个 变量 ， 无 论 
值 为 什么 ， 都 表示 用 户 选中 了 Is this a favorite? 复 选 框 。 如 果 用 户 没 有 选中 它 ， 就 不 会 设置 这 个 变 


_ 国 . 
于， 




















那 就 会 将 0 赋 给 $favorite 变 量 。 
(7) 定义 并 执行 查询 : 
$query = "INSERT INTO quotes (gquote, source, favorite) VALUES ('S$quote', 


'$Ssource', Sfavorite)"; 
$r = mysql_query ($query, $dbc); 


这 个 查询 使 用 已 经 定义 的 变量 分 别 为 3 个 字段 指定 了 值 。 程 序 会 自动 给 其 他 两 个 字段 





(quote_igd 和 date_entered) 赋值 ， 这 是 在 定义 表 的 时 候 设 置 的 。 


4PAo 


(8) 根据 结果 打印 相应 信息 : 


if (mysql_ affected rows($dbc) == 1) { 
print '<p>Your quotation has been stored.</p>'; 
} else { 
print '<p class="error">Could not store the quote because: <br/>' 
mysql_error ($dbc) .'.</p><p>The query being run was: ' . $query . '</p>'; 
} 


如 果 INSERT 查 询 在 数据 库 中 添加 了 一 个 新 行 ， 就 会 显示 “Your quotation has been stored.” 信 
否则 ， 就 会 显示 调试 信息 ， 你 可 以 从 中 查找 错误 原因 。 
(9) 完成 验证 条 件 语 句 : 

】 else { // 没有 填写 名 人 名 言 


print '<p class="error">Please enter a quotation and a source!</p>'; 


} 
(10) 完成 提交 条 件 语句 和 PHP 代 码 : 
】 // 结束 提交 条 件 语 向 。 


?> 


(11) 创建 表单 


<form action="add quote.php" method="post"> 
<p><label>Quote <textarea name="quote" rows="5" cols="30"></textarea> 
</label></p> 
<p><label>Source <input type="text" name="source" /> </label></p> 
<p><label>Is this a favorite? <input type="checkbox" name="favorite" /></label></p> 
<p><input type="submit" name="submit" value="Add This Quote!" /></p> 
</form> 


这 个 表单 由 一 个 文本 区 、 一 个 文本 输入 框 和 一 个 复 选 





[HH 


(当然 还 包含 提交 按钮 ) 组 成 。 表 单 








现在 不 是 一 个 粘性 表单 ， 你 可 以 自己 加 入 这 个 特性 。 


(12) 包含 页 脚 文件 : 


<?php include('templates/ footer.html'); ?> 
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(13) 将 文件 保存 为 ad9_quote .php， 并 在 Web 浏 览 器 中 测试 (参见 图 13-13)。 








My Site of Quotes 


Add a Quotation 




















La 








13-13 ”添加 名 人 名 言 后 的 页 面 





w 提示 
口 新 建 (INSERT)、 检 索 (SELECT)、 更 新 和 删除 数据 库 记 录 的 功能 简写 为 CRUD (Create、 
Retrieve、Update、Delete ) 。 


13.8 显示 名 人 名 言 


网 站 的 管理 部 分 需要 有 一 个 页 面 能 够 列 出 所 有 存储 在 数据 库 中 的 名 人 名 言 (参见 图 13-14) 。 
虽然 这 个 脚本 也 可 以 用 于 公开 的 页 面 ,但 它 的 主要 目的 是 为 管理 员 提 供 一 些 快捷 链接 ,以 便 编辑 、 
删除 那些 名 言 。( 这 不 同 于 在 公开 页 面 随机 搜索 某 个 名 言 来 进行 管理 。) 

与 add_quote.php 脚 本 类 似 ， 这 个 页 面 也 只 允许 管理 员 访 问 。 











My Site of Quotes 


All Quotes 


)uote Admin: Edit Delete 








dmin: Edit Delete 


In ai irs if's a healthy thing n 
hang Westion mark on the thin 


图 13-14 ”所 有 名 人 名 言 的 列表 ， 每 个 名 人 名 言 下 都 有 编辑 和 删除 链接 
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瀛 创 建 view_quotes .php| 











(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 PHP 文 档 ， 命 名 为 view_quotes .php (参看 脚本 13-8): 


<?php // 脚本 13-8 - view_quotes.php 
define('TITLE', 'View All Quotes'); 
include('templates/header.html'); 
print '<h2>All Quotes</h2>'; 











脚本 13-8 这 个 脚本 显示 当前 存储 的 所 有 名 人 名 言 ， 并 为 管理 员 提供 了 编辑 和 删除 链接 


<?php // 脚本 13-8 - view_quotes.php 
/* 脚本 显示 所 有 名 人 名 言 。*/ 


// 包含 页 头 文件 : 
define('TITLE', 'View All Quotes'); 
include('templates/header.html'); 


print '<h2>All Quotes</h2>'; 


‘OoO 和 OOOOPODODP 


10 // 强制 只 有 管理 员 可 以 访问 该 页 面 : 

11 if (!is_ administrator()) { 

12 print '<h2>Access Denied!</h2><p class="error">You do not have permission to 
access this page.</p>'; 

include('templates/footer.html'); 

exit(); 


// 包含 数据 库 连 接 : 


3 
14 
上 上 5， 计 
6 
7 
8 include('../mysql_connect.php'); 





20  // 定义 查询 : 
21 S$query = 'SELECT quote id, quote, source, favorite FROM quotes ORDER BY date entered DESC'; 














23 // 运行 查询 : 
24 if (Sr = mysql_ query ($query, S$dbc)) { 





2:5 

26 // 返回 查询 结果 : 

27 while ($row = mysql_ fetch array ($sr)) { 

28 

29 // 打印 查询 结果 : 

30 print "<div><blockquote>{$row['gquote']}</blockquote>- {$row['source']}\n"; 

31 

32 // 判断 是 否 为 受 欢 迎 的 。 

33 if ($row['favorite'] == 1) { 

34 print ' <strong>Favorite!</strong>'; 

35 3 

36 

37 // 添加 管理 员 链 接 : 

38 print "<p><b>Quote Admin:</b> <a href=\"edit quote.php?id={$row['quote id']}\"> 
Edit</a> <-> 

39 <a href=\"delete quote.php?id={$row['gquote_ id']}\">Delete</a></p></div>\n"; 

40 


41 } // 结束 循环 。 
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42 

43 } else { // 没有 运行 查询 。 

44 print '<p class="error">Could not retrieve the data because:<br/>' . mysql error 
(Sdbc) . '.</p><p>The query being run was: ' . Squery . '</p>'; 

45 } // 结束 查询 条 件 语 向 。 

46 

47 mysql_close($dbc); // 关闭 连接 。 

48 

49 include('templates/footer.html'); // 包含 页 脚 文 件 。 

50 ?> 

(2) 如 果 用 户 不 是 管理 员 ， 则 终止 脚本 : 

if (!is administrator()) { 


print '<h2>Access Denied!</h2> <p class="error">You do not have permission to access 
一 this page.</p>'; 

include('templates/footer.html'); 

exit(); 


这 段 代码 几乎 和 aqaq_quote.php 中 的 代码 完全 一 样 ， 只 是 修改 了 浏览 器 上 显示 的 标题 。 非 
管理 员 用 户 会 像 上 一 节 一 样 被 拒绝 访问 (参见 图 13-15)。 


<!-—- BEGIN CHANGEABLE CONTENT. --><h2>All Quo 
<p><h>Quote Admin:</h> <a href="edit quote.php?id=13">Edit</a. 
<a href="delete dquote.php?id=13">Delete</a></ 
<div><hlockquote>It is the mark of an educated mind to be abl 
<strong>Favorite!</strong><p><h>Quote Admin:</h> <a href="ed| 
<a href="delete quote.php?id=9">Delete</a></p 
<div><hblockquote>In all affairs it's a healthy thing now and 
mark on the things you have long taken for granted.</hlockquo 
<p><h>Quote Admin:</h> <a href="edit quote.php?id=5">Edit</a> 
<a href="delete dquote.php?id=5">Delete</a></p 
<qdiv><hlockquote>Nurture your mind with great thoughts, for vy 
<p><h>Quote hdmin:</h> <a href="edit quote.php?id=6">Edit</a> 
<a href="delete quote.php?id=6">Delete</a></p 


图 13-15 ”该 页 面 的 HTML 源 代码 显示 了 quote_id 的 值 是 如 何 通过 URL 被 发 送 到 连接 页 面 的 
(3) 包含 数据 库 连接 并 定义 查询 : 


include('../mysql_connect.php'); 
$query = 'SELECT quote _ id, quote, source, favorite FROM quotes ORDER BY 
date_entered DESC ' ; 


这 个 查询 返回 数据 库 中 所 有 记录 的 4 列 (quote_id、quote、source、favorite)， 返 回 
结果 按 数据 添加 日 期 降序 显示 。 
(4) 执行 查询 并 获取 结果 : 


if ($r = mysql_ query ($query, .$dbc)) { 
while ($row = mysql_ fetch array($r)) { 


while 循 环 代码 在 第 12 章 阐释 过 ， 不 过 当时 并 没有 使 用 这 段 代 码 。 这 段 代码 的 目的 是 获取 所 
有 查询 语句 返回 的 记录 。 


(5) 开始 打印 记录 : 
print "<div><blockquote> {$row['gquote']}</blockquote>- {$row['source']}\n"; 


TT 
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代码 以 <aiv> 标 签 开始 ， 将 名 人 名 言 放 在 一 对 <blockquote> 标 签 之 中 ， 随 后 显示 名 言 的 
出 处 。 
(6) 判断 名 言 是 否 受 欢迎 : 


if ($row['favorite'] == 1) { 
print ' <strong>Favorite! </strong>'; 


} 

$row['favorite'] 的 值 可 以 是 1 或 9。 如 果 是 1， 会 在 名 言 后 面 显示 一 个 明显 的 Favorite! 
标识 。 

(7) 添加 管理 链接 ， 用 二 编辑 和 删除 名 人 名 言 : 


print "<p><b>Quote Admin:</b><a href=\"edit_ quote.php?id= 
{$row['quote_id']}\">Edit</a> <-> 
<a href=\"delete quote.php?id= {$row['quote_ id']}\">Delete</a> </p></div>\n"; 


必须 为 每 个 名 言 创建 两 个 链接 。 第 一 个 连接 到 edit_quote.php， 第 二 个 连接 到 delete_ 
quote .php。 每 个 链接 必须 通过 URL 发 送 quote_id， 类 似 第 12 章 示例 中 的 代码 。 
(8) 完成 while 循 环 和 mysal_query () 条 件 语句 : 


} // 结束 循环 。 
} else { // 没有 运行 查询 。 
print '<p class="error">Could not retrieve the data because:<br/>' . mysql_error 
($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 
】 // 结束 查询 条 件 语句。 


(9) 断 开 数 据 库 连接 : 
mysql_close (sdbc); 
(10) 完成 页 面 : 


include('templates/footer.html'); 
全 


(11) 将 文件 保存 为 view_quotes .php 并 在 Web 浏 览 器 中 测试 。 








Vv 提示 
口 本 章 的 一 些 查 询 语句 可 以 使 用 ORDER BY 子 句 按 某 个 列 (或 值 ) 排序 ， 查 询 语句 没有 选择 
的 列 也 可 以 作为 排序 依据 。 


13.9 ”编辑 名 人 名 言 


view_cuotes .php 页 面 (和 后 面 要 创建 的 index. php 页 面 ) 包含 指向 edit_quote.php 的 
链接 ， 用 于 管理 员 编 辑 名 人 名 言 。 从 功能 上 看 ,这 个 脚本 与 第 12 章 创建 的 edit_entry .php 非 常 
类 似 : 

(1) 脚本 需要 在 URL 中 接收 ia 值 ， 

(2) 通过 接收 的 ia 值 获取 记录 并 弹出 表单 (参见 图 13-16)，; 
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(3) 表单 提交 后 ， 会 验证 表单 数据 〈 只 实现 最 基本 的 验证 )， 

(4) 如 果 数 据 通过 验证 ， 就 会 更 新 数据 库 中 的 相关 记录 。 

本 市 中 的 代码 大 部 分 都 在 之 前 的 章 市 曾 释 过 。 你 需要 学 习 的 新 东西 是 如 何 根据 原 有 的 值 选中 
或 不 选中 表单 的 复 选 框 元 素 。 








My Site of Quotes 


Edit a Quotation 
It is the mark of an educated 
mind to be able to entertain a 
thought without accepting it. 





|Aristotle 




















图 13-16 表单 元 素 会 包含 数据 库 记 录 中 的 值 ， 并 同 表单 一 起 弹出 


这 创建 edit_quote .php 
(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 PHP 文 档 ， 命 名 为 edit_quote .php (参看 脚本 13-9): 


<?php // 脚本 13-9 -edit_quote.php 
define('TITLE', 'Edit a Quote'); 
include('templates/header.html'); 
print '<h2>Edit a Quotation</h2>'; 




















脚本 13-9 edit_quote .php 脚 本 用 于 让 管理 员 编 辑 已 有 记录 
<?php // 脚本 13-9 - edit_quote.php 
/* 脚本 编辑 名 人 名 言 。*/ 


define('TITLE', 'Edit a Quote'); 
include('templates/header.html'); 











1 
2 
3 
4  // 定义 页 面 标题 并 包含 页 头 文件 : 
5 
6 
7 
8 





print '<h2>Edit a Quotation</h2>'; 


9 

10 // 强制 只 有 管理 员 可 以 访问 该 页 面 : 

11 if (!is administrator()) { 

耳光 print '<h2>Access Denied!</h2> 


<p class="error">You do not have permission to access this page.</p>'; 


.3 include('templates/footer.html'); 
下 52 让 
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16 
17 // 包含 数据 库 连 接 : 
18 include('../mysql_ connect.php'); 
19 
20 if (isset($_GET['id']) && is numeric($_GET['id']) && ($_GET['id'] > 0) ) 
{ // Display the entry in a form: 
21 
22 // 定义 查询 。 
23 $query = "SELECT quote, source, favorite FROM quotes WHERE quote id={$_GET['id']}"; 
24 if (Sr = mysql_query ($query，Sdbc)) { // 运行 查询 。 
25 
26 Srow = mysql_fetch array (S$r); // 返回 信息 。 
2 
28 // 创建 表单 : 
29 print '<form action="edit quote.php" method="post"> 
30 <p><label>Quote <textarea name="quote" rows="5" cols="30">' 
htmlentities($row['quote']) . '</textarea></label></p> 
31 <p><label>Source <input type="text" name="source"value="' . htmlentities 
(Srow['source']) . '" /></label></p> 
32 <p><label>Is this a favorite? <input type="checkbox" name="favorite" 
value="yes"'; 
33 
34 // 检查 是 否 选中 受 欢迎 复 选 框 : 
35 if ($row['favorite'] == 1) { 
36 print ' checked="checked"'; 
B33 } 
38 
39 // 完成 表单 : 
40 print ' /></label></p> 
41 <input type="hidden" name="id" value="' . $_GET['id'] . '" /> 
42 <p><input type="submit" name="submit" value="Update This Quote!" /></p> 
43 </form>'; 
44 
45 } else { // 无 法 获取 信息 。 
46 print '<p class="error">Could not retrieve the quotation because:<br/>' 
mysql_error($dbc) . '.</p><p>The query being run was: ' . Squery . '</p>'; 
47 } 
48 
49 } elseif (isset($_ POST['id']) && is numeric($ POST['id']) && ($_ POST['id'] > 0)) 
{ // 处 理 表单 。 
50 
51 // 验证 表单 数据 并 确保 安全 : 
52 $sproblem = FALSE; 
53 if ( lI!empty($_POST['gquote']) && !empty($_POST['source']) ) { 
54 
55 // 准备 查询 中 使 用 的 值 : 
56 $Squote = mysql_real_ escape string(trim(strip_ tags($_ POST['quote'])), $dbc); 
5:7 SSsource = mysql_ real escape string(trim(strip tags($_POST['source'])), $dbc); 
58 
59 // 创建 favorite 值 : 
60 if (isset($_POST['favorite'])) { 
61 $favorite = 1; 
62 } else { 
63 Sfavorite = 0; 
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64 } 

65 

66 } else { 

67 print '<p class="error">Please Submit both a quotation and a source.</p>'; 

68 sproblem = TRUE; 

69 } 

70 

71 if (!S$Sproblem) { 

72 

73 // 定义 查询 。 

74 Squery = "UPDATE quotes SET quote='$quote', source='$source', favorite= 
Sfavorite WHERE quote_ id={$_POST['id']}"; 

75 if (Sr = mysql query ($query, S$dbc)) { 

76 print '<p>The quotation has been updated.</p>'; 

77 } else { 

78 print '<p class="error">Could not update the quotation because:<br/>' 

mysql_error($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 

79 } 

80 

81 } // 一 切 正常 ! 

82 

83 }else { // 没有 获取 id。 

84 print '<p class="error">This page has been accessed in error.</p>'; 

85 } // 结束 主 条 件 语 向。 

86 

87 mysql_close($dbc); // 关闭 连接 。 

88 

89 include('templates/footer.html'); // 包含 页 脚 文 件 。 

90. 32> 

(2) 如 果 用 户 不 是 管理 员 则 终止 脚本 : 

if (!is administrator()) { 


print '<h2>Access Denied!</h2> <p class="error">You do not have permission to access 
this page.</p>'; 
include('templates/footer.html'); 


exit(); 
} 
(3) 包含 数据 库 连 接 : 
include('../mysql_connect.php'); 


脚本 中 的 两 个 功能 (显示 和 处 理 表单 )， 都 需要 数据 库 连 接 ， 所 以 这 里 包含 数据 库 连 接 文件 。 

(4) 验证 URL 是 否 接 收 到 数值 型 ia 值 : 

if (isset($_GET['id']) && is numeric($_ GET['id']) && ($_GET['id'] > 0) ) { 

这 个 条 件 语句 和 第 12 章 的 类 似 ， 除 此 之 外 还 会 检查 id 值 是 否 大 于 0。 添 加 这 个 子 句 是 出 于 脚 
本 安全 性 的 考虑 : is_numeric () 测 试 查询 中 的 值 是 否 安全 , 如 果 ia 值 不 可 用 , 就 不 会 执行 查询 。 

(5) 定义 并 执行 查询 : 


Squery = "SELECT quote, source, favorite FROM quotes WHERE quote_ id={$_GET['id']}"; 
if ($r = mysql_query ($query, .$dbc)) { | 


Srow = mysql_fetch array ($r); 
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查询 返回 记录 中 的 3 列 。 
(6) 创建 表单 : 


print '<form action="edit quote.php" method="post"> 
<p><label>Quote <textarea name="quote" rows="5" cols="30">' 





"htmlentities($row['quote']) .'</textarea></label></p> 
<p><label>Source <input type="text" name="source"value="' 
—htmlentities($row['source']) .'" /></label></p> 


<p><label>Is this a favorite? <input type="checkbox" 
"name="favorite" value="yes"'; 


表单 会 返回 到 同一 个 页 面 。 表单 的 开头 部 分 是 文本 区 , 它 的 值 是 事先 从 数据 库 中 检索 出 来 的 
名 人 名 言 。 这 个 值 会 通过 htmlentities () 函数 进行 转换 ， 以 确保 安全 。 

接 下 来 , 创建 了 一 个 文本 输入 框 ， 它 的 值 是 事先 从 数据 库 中 检索 出 的 名 人 名 言 出 处 。 最 后 是 
创建 是 否 受 欢迎 的 复 选 框 。 注意 , 这 个 复 选 框 目前 还 没有 完成 , 接 下 来 的 第 (7) 步 脚本 就 会 检查 复 
选 框 是 否 被 选中 。 

(7) 检查 是 否 选中 受 欢迎 复 选 框 : 


if ($row['favorite'] == 1) { 
print ' checked="checked"'; 
} 


如 果 数 据 库 中 favorite 的 值 等 于 1, 就 说 明 管 理 员 先 前 选中 过 受 欢 迎 复 选 框 ,在 这 种 情况 下 ， 
需要 添加 一 些 HTML， 将 复 选 框 状态 置 为 选中 。 在 这 有 段 代码 之 后 ， 受 欢迎 复 选 框 的 后 台 HTML 代 
码 可 能 是 : 

<input type="checkbox" name="favorite" value="yes" 
或 (参见 图 13-17) 


<input type="checkbox" name="favorite" value="yes" checked="checked" 

















<textarea name="duote" rows="5" ColLs="30">IL is the mark of an educated mind to be ab 
<input type="text" name="source"value="Aristotle" jf></labhel></p> 

s a favorite? <input type="checkbox" name="favorite" value="yes" checked="checked" /> 
en name="id" value="9" /> 


图 13-17 ”该 页 面 的 HTML 源 代码 ， 在 首次 访问 时 ， 显 示 先 前 选中 的 受 欢迎 复 选 框 
(8) 完成 表单 : 


print ' /></label></p> 



































<input type="hidden" name="id" value="' . $_GET['id'] . '" /> 
<p><input type="submit" name="submit" value="Update This Quote!" /></p> 
</fOrm>" > 


print 语 句 首先 关闭 复 选 框 。 接 下 来 ,表单 必须 将 ia 值 存储 在 隐藏 的 文本 框 中 ， 以 便 在 提交 
表单 时 使 用 。 
(9) 如 果 没 有 检索 到 记录 ， 提 示 错 误 信息 : 


} else { // 无 法 获取 信息 。 
print '<p class="error">Could not retrieve the quotation 
» because:<br/>' . mysql_ error($dbc) . '.</p><p>The 
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一 query being run was: ' . $query . '</p>'; 


} 
(10) 检查 表单 提交 情况 : 
} elseif (isset($_POST['id']) && is numeric($_ POST['id']) && ($_POST['id'] > 0)) { 
这 个 条 件 语 句 是 脚本 第 二 个 部 分 (处理 表单 提交 ) 的 起 始 处 。 验 证 的 方式 与 第 (4) 步 类 似 , 但 
这 里 引用 的 是 $_POST['id']， 而 不 是 $_GET['id']。 
(11) 验证 表单 数据 : 


$sproblem = FALSE; 
if ( !empty($_POST['quote']) &&!empty($_POST['source']) ) { 


表单 验证 的 方式 同 ad9_quote .php 脚 本 的 验证 方式 相同 。 
(12) 准备 值 以 便 在 查询 中 使 用 : 


$Squote = mysql_real escape _ string (trim(strip_ tags($_POST['gquote'])), $dbc); 
$source = mysql_ real escape_ string(trim(strip tags($_POST['source'])), $dbc); 
if (isset($_POST['favorite'])) { 

$sfavorite = 1; 
} else { 


Sfavorite = 0; 


} 


这 段 代码 同 add_quote .php 脚 本 里 使 用 的 相同 。 
(13) 如 果 表 单 没有 完成 ， 显 示 错 误 信息 : 
} else { 
print '<p class="error">Please submit both a quotation and a source.</p>'; 


$sproblem = TRUE; 
} 


(14) 如 果 没 有 问题 发 生 ， 更 新 数据 库 : 


if (!S$Sproblem) { 
$query = "UPDATE quotes SET quote='S$Squote', source='$source', favorite=$favorite 
一 WHERE quote_id={$_POST['id']}"; 
if (Sr = mysdql_duery(Sduery，S$qbc)) { 
print '<p>The quotation has been updated.</p>'; 


查询 更 新 每 条 记录 的 3 列 值 。 两 个 字符 串 值 用 单 引 号 引起 来 (在 查询 内 ) ， 而 $favorite 是 数 
值 型 ， 不 需要 使 用 单 引 号 。 

WHERE 子 句 指明 要 更 新 的 记录 ， 是 最 重要 的 部 分 。 

最 后 ， 打 印 更 新 成 功 的 信息 (参见 图 13-18 ) 。 

















My Site of Quotes 


Edit a Quotation 














WA 











13-18 成 功 编辑 记录 后 的 页 面 
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(15) 如 果 更 新 失败 ， 显 示 错 误 信 息 : 


} else { 
print '<p class="error">Could not update the quotation because:<br/>' . mysql_ 
» error($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 
} 
区 
(16) 完成 条 件 语句 : 
} // 一 切 正常 ! 


} else { // No ID set. 


print '<p class="error">This page has been accessed in error.</p>'; 


}】 // 结束 主 条 件 语句 。 

else 子 句 处 理 没 有 通过 GET 或 POST 方式 收 到 有 效 iq 值 的 情况 。 
(17) 关闭 数据 库 连接 并 完成 页 面 : 

mysal_close ($dbc); 


include('templates/footer.html'); 


?> 


(18) 保存 文件 并 在 Web 训 览 器 中 测试 ( 单 击 view_quotes .php 上 的 链接 )。 


13.10 ”删除 名 人 名 言 








删除 已 有 名 人 名 言 的 脚本 类 似 第 12 章 的 delete_entry .php 脚 本 。 第 一 次 访问 时 , 如 果 通 过 


URL 传 递 的 是 一 个 有 效 的 19， 就 会 显示 要 删除 的 名 人 名 言 记录 (参见 








图 13-19)。 如 果 管 理 员 单 击 





了 提交 按钮 ， 表 单 会 返回 相同 的 页 面 ， 并 从 数据 库 中 删除 这 条 记录 (参见 图 13-20)。 





My Site of Quotes 


Delete a Quotation 

















Delete a Quotation 


























党 创建 delete_quote .php 


图 13-19 ”删除 记录 的 第 一 步 是 确认 要 删除 的 记录 图 13-20 表单 提交 后 ， 名 人 名 言 会 被 


删除 ， 同 时 打印 成 功 信息 








(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 PHP 文 档 , 命名 为 delete_quote.php (参看 脚本 13-10): 








<?php // 脚本 13-10 - delete_quote.php 
define('TITLE', 'Delete a Quote'); 
include('templates/header.html'); 
print '<h2>Delete a Quotation</h2>'; 
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脚本 13-10 ”delete_quote.php 脚 本 用 于 让 管理 员 删 除 已 有 记录 


<?php // 脚本 13-10 - delete_quote.php 


‘OmoJOAOUURODP 





DI 
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/* 


Yeh 


脚本 删除 名 人 名 言 。*/ 


定义 页 面 标题 并 包含 页 头 文件 : 


define('TITLE', 'Delete a Quote'); 
include('templates/header.html'); 





print '<h2>Delete a Quotation</h2>'; 


// 
if 


} 


// 


强制 只 有 管理 员 可 以 访问 该 页 面 : 

(!is_agdministrator()) { 

print '<h2>Access Denied!</h2> 

<p class="error">You do not have permission to access this page.</p>'; 
include('templates/footer.html'); 

et 


包含 数据 库 连 接 : 


include('../mysql_connect.php'); 


if 
// 


{ // 处 理 表单 。 


} elseif (isset($_POST['id']) && is_numeric(S_POST['idq']) && ($_POST['id'] > 0) ) 13 


// 定义 查询 : 
$query = "SELECT quote, source, favorite FROM quotes WHERE quote id= 








(isset($_GET['id']) && is numeric($_GET['id']) && ($_GET['id'] > 0) ) { 
在 表单 中 显示 名 人 名 言 : 














{SGET[ Td yy" 


if (S$r = mysql_query($Squery，S$dbc)) { // 运行 查询 。 


Srow = mysql_fetch array($r); // 返回 信息 。 


// 创建 表单 : 

print '<form action="delete quote.php" method="post"> 

<p>Are you sure you want to delete this quote?</p> 

<div><blockquote>' . $row[l'quote'] . '</blockquote>- ' . $row['source']; 


// 检查 是 否 选 中 受 欢迎 复 选 框 : 
if ($row['favorite'] == 1) { 
print ' <strong>Favorite!</strong>'; 





print '</div><br/><input type="hidden" name="id" value="' . $_GET['id'] . '" /> 
<p><input type="submit" name="submit" value="Delete this Quote!" /></p> 
</EOrMmS 


} else { // 无 法 获取 消息 。 
print '<p class="error">Could not retrieve the quote because:<br/>' 
mysql_error($dbc) . '.</p><p>The query being run was: ' . Squery . '</p>'; 
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48 // 定义 查询 : 

49 Squery = "DELETE FROM quotes WHERE quote id={$_POST['id']} LIMIT 1"; 

50 Sr = mysql_query ($query，Sdbc); // 执行 查询 。 

5 

52 // 打印 结果 : 

53 if (mysql_affected rows($dbc) == 1) { 

54 print '<p>The quote entry has been deleted.</p>'; 

55 Ese 

56 print '<p class="error">Could not delete the blog entry because:<br/>' 
mysql_error($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 

57 } 

58 

59 } else { // 没有 获取 id。 

60 print '<p class="error">This page has been accessed in error.</p>'; 

61 } // 结束 主 条 件 语 向。 

62 

63 mysql_close($dbc); // 关闭 连接 。 

64 

65 include('templates/footer.html'); 

66 ?> 

(2) 如 果 用 户 不 是 管理 员 则 终止 脚本 : 

if (!is administrator()) { 


print '<h2>Access Denied!</h2> <p class="error">You do not 
have permission to access this page.</p>'; 

include('templates/footer.html'); 

exit(); 


} 

(3) 包含 数据 库 连 接 : 

include('../mysql_connect.php'); 

(4) 验证 DD 值 是 从 URL 中 接收 的 : 

if (isset($_GET['id']) && is numeric($_GET['id']) && ($_GET['id'] > 0) ) { 
这 个 条 件 语句 与 edit_quote.php 中 使 用 的 一 样 。 

(5) 检索 要 删除 的 数据 : 


$query = "SELECT quote, source, favorite FROM quotes WHERE quote_ id={$_GET['id']}"; 
if (Sr = mysql_ query ($query, S$dbc)) { Srow = mysql_fetch array (S$r); 


从 数据 库 中 返回 检索 记录 的 3 个 字段 。 因 为 这 个 脚本 只 会 处 理 一 条 记录 ， 所 以 只 在 循环 外 调 
次 mysql_fetch_array () 半数 。 
(6) 开始 创建 表单 : 


print '<form action="delete quote. php" method="post"> 
<p>Are you sure you want to delete this quote?</p> 




















<div><blockquote>' . $row[l'gquote'] .'</blockquote>- ' 
$row['source']; 
表单 主要 用 于 显示 名 人 名 言 。 


(7) 如 果 名 人 名 言 是 受 欢 迎 的 ， 显 示 Favorite! 标 记 : 
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if (Srow['favorite'] == 1) { 
print ' <strong>Favorite! </strong>'; 
} 
这 段 代码 与 view_quotes .php 页 面 中 的 一 样 ， 显 示 这 是 受 欢 迎 的 名 言 。 
(8) 完成 表单 : 
print '</div><br/> 
<input type="hidden" name="id" value="' . $_GET['id'] . '" /> 
<p><input type="submit" name="submit" value="Delete this Quote!" /></p> 
</form>'; 


表单 必须 包含 隐 含 的 输入 框 ， 在 提交 时 ， 这 个 输入 框 会 将 名 言 的 id 传 回 页 面 。 
(9) 完成 mysql_query () 条 件 语句 : 


} else { // 无 法 获取 信息 。 
print '<p class="error">Could not retrieve the quote 
一 because:<br/>' . mysql_ error($dbc) . '.</p><p>The 
"query being run was: ' . $query . '</p>'; 


} 

如 果 查 询 失 败 、MySQL 出 错 或 查询 本 身 出 错 ， 则 会 显示 调试 建议 。 

(10) 检查 表单 提交 情况 : 

} elseif (isset($_ POST['id']) && is numeric($ POST['id']) && ($_ POST['id'] > 0) )) { 

这 个 条 件 语句 开启 了 脚本 的 第 二 个 部 分 (处 理 表 单 提交 )。 验 证 方式 和 第 (4) 步 相同 ， 但 是 这 
里 引用 的 是 $_POST['id']， 而 不 是 $_GET['id']。 

(11) 删除 记录 : 


$query = "DELETE FROM quotes WHERE quote_ id={$_POST['id']} LIMIT 1"; 
Sr = mysql_ query ($query, S$dbc); 


DELETE 语 名 会 删除 记录 。wHERE 子 句 指 明 要 删除 的 特定 记录 ，LIMIT 1 子 句 是 另 一 个 限制 条 













































































件 5 
(12) 报告 结果 : 
if (mysql_affected rows($dbc) == 1) { 
print '<p>The quote entry has been deleted.</p>'; 
} else { 
print '<p class="error">Could not delete the blog entry because:<br/>' . mysql_ 
error($dbc) . '.</p><p>The query being run was: ' . $query . '</p>'; 
} 
如 果 语 句 执行 成 功 ， 就 会 删除 一 条 记录 ， 并 向 用 户 显示 信息 (参见 图 13-20)。 
(13) 完成 条 件 语句 : 


) else { // 没有 获取 id。 
print '<p class="error">This page has been accessed in error.</p>'; 
】 // 结束 主 条 件 语 向 。 


else 子 名 处 理 没 有 通过 GET 或 POST 方式 收 到 有 效 iq 值 的 情况 。 
(14) 关闭 数据 库 连 接 并 完成 页 面 : 
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mysql_close (sdbc); 
include('templates/footer.html'); 
他 


(15) 保存 文件 并 在 Web 浏 览 器 中 测试 ( 单 击 view_quotes .php 上 的 链接 )。 
13.11 创建 主页 


ek 我 们 需要 创建 主页 。 对 于 这 个 网 站 来 说 ,主页 面 是 唯一 一 个 面向 公众 的 页 面 。 主 页 会 





一 条 名 人 名 言 ， 这 个 名 言 可 以 是 以 下 几 种 类 型 : 
口 最 新 添加 的 名 人 名 言 (默认 )， 
口 随机 的 名 人 名 言 
口 随机 的 受 欢 迎 的 名 人 名 言 。 
要 实现 这 个 效果 ， 链 接 将 在 URL 中 传送 不 同 的 值 到 这 个 脚本 (参见 图 13-21)。 


























如 果 用 户 是 管理 员 , 这 个 脚本 还 需要 在 当前 显示 的 名 人 名 言 下 面 显示 管理 链接 (编辑 链接 和 





删除 链接 ) (参见 图 13-22)。 


tes - Mozilla Firefox 
tes - Mozilla Firefox 
Cay http://10,0,1,2:8888s/index,.php?favorite=true 他 
{ay http://10.0.1.,2:8888/index,php?random=true 3 了 


jE My Site of Quotes 
My Site of Quotes 


























Edit Delete 
图 13-21 通过 URL 传 送 不 同 的 值 以 图 13-22 当 管 理 员 查看 主页 时 , 会 显示 
显示 不 同 的 名 言 更 多 管理 链接 





过 创建 index .php 














(1) 在 文本 编辑 器 或 IDE 中 新 建 一 个 PHP 文 档 ， 命 名 为 ndex.php (参看 脚本 13-11 


<?php // 脚本 13-11 - index.php 





脚本 13-11 网 站 的 主页 一 次 显示 一 条 名 人 名 言 ， 在 管理 员 登 录 时 会 显示 管理 链接 


<?php // 脚本 13-11 - index.php 
/* 这 是 网 站 的 主页 。 包 含 以 下 内 容 : 
- 最 新 名 人 名 言 (默认 ) 

- 或 ， 随 机 名 人 名 言 

- 或 ， 随 机 受 欢迎 的 名 人 名 言 */ 


// 包含 页 头 文件 : 
include('templates/header.html'); 


OARODP 


) : 
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// 包含 数据 库 连 接 : 
include('../mysql_connect.php'); 


// 定义 查询 …… 

// 根据 URL 传 递 过 来 的 值 更 改 查 询 方 式 : 

If (isset($_GET['random'])) { 
$query = 'SELECT quote id, quote, source, favorite FROM quotes ORDER BY RAND() 
DESG TMIT TL 

} elseif (isset($ GET['favorite'])) { 
$query = 'SELECT gquote_id, quote, source, favorite FROM quotes WHERE favorite=1 
ORDER BY RAND() DESC LIMIT 1'，; 

} else { 
$query = 'SELECT quote_ id, quote, source, favorite FROM quotes ORDER BY 
date_entered DESC LIMIT 1'，; 


























} 


// 运行 查询 : 
if (Sr = mysql_ query ($query, $dbc)) { 


// 返回 查询 结果 : 
$row = mysql_fetch array (S$r); 


// 打印 查询 结果 : 
print "<div><blockquote>{$row['quote']}</blockquote>- {$row['source']}"; 


// 判断 是 否 为 受 欢 迎 的 ? 
if (Srow['favorite'] == 1) { 
print ' <strong>Favorite!</strong>'; 


// 完成 div 标 签 : 
Deine. /iv 


// 如 果 管 理 员 登录 ， 显 示 管 理 员 链接 : 
if (is administrator()) { 
print "<p><b>Quote Admin:</b> <a href=\"edit quote.php?id={$row['quote id']} 
\">Edit</a> <-> 
<a href=\"delete quote.php?id={$row['quote_ id']}\">Delete</a> 
/DN 


} else { // 没有 运行 查询 。 
print '<p class="error">Could not retrieve the data because:<br/>' 
mysql_error($dbc) . '.</p><p>The query being run was: ' . Squery . '</p>'; 
} // End of query IF. 





mysql_close ($dbc); // 关闭 连接 。 


print '<p><a href="index.php">Latest</a> <-> <a href="index.php?random=true"> 
Random</a> <-> <a href="index.php?favorite=true">Favorite</a><p>'; 


include('templates/footer.html'); // 包含 页 脚 文 件 。 


归 六 
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(2) 包含 头 文件 : 
include('templates/header.html'); 


主页 不 需要 自 定义 标题 ， 因 此 在 包含 头 文件 之 前 不 需要 定义 常量 。 


(3) 包含 数据 库 连 接 : 
include('../mysql_connect.php'); 
(4) 开始 定义 要 运行 的 查询 : 
if (isset($_GET['random'])) { 
Squery = 'SELECT quote _ id, quote, source, favorite FROM 





"quotes ORDER BY RAND() DESC LIMIT 1'; 


如 果 $_GETI['random' ] 变 量 已 设置 ， 用 户 单 击 某 个 链接 请 求 一 条 随机 名 人 名 言 。 无 论 这 个 




















变量 的 值 是 什么 ， 只 要 设置 了 即 可 。 


所 有 的 查询 都 会 返回 来 自 同一 行 的 4 个 列 值 (quote_id、quote、source 和 favorite)。 





如 果 只 检索 一 行 ， 使 用 LIMIT 1 子 句 即 可 。 


RANIj 





使 用 ORDER BY RAND () 子 句 ， 在 MySQL 中 选择 一 个 随机 的 行 。 这 个 代码 使 用 MySQL 的 








合 中 


间 的 


D () 函数 ，RAND 是 random 的 缩写 。 这 个 查询 首先 以 随机 顺序 选择 所 有 记录 ， 之 后 返回 这 个 集 








的 首 条 记录 。 
(5) 定义 查询 ， 该 查询 选择 一 条 随机 的 受 欢迎 的 记录 : 
} elseif (isset($ GET['favorite'])) { 
Squery = 'SELECT quote_ id, quote, source, favorite FROM 


"quotes WHERE favorite=1 ORDER BY RAND() DESC LIMIT 1'; 


这 个 查询 与 第 (4) 步 的 一 样 ， 但 这 里 使 用 wHERE 子 句 来 选择 那些 favorite 值 等 于 1 的 记录 。 
(6) 定义 默认 查询 ; 
} else { 


Squery = 'SELECT quote_ id, quote, source, favorite FROM 
quotes ORDER BY date_ entered DESC LIMIT 1'，; 

















} 

如 果 没 有 通过 URL 传 送 任 何 值 , 主页 会 显示 最 新 添加 的 名 人 人 名言。 实现 方法 是 按 名 言 添 加 时 
降序 排序 所 有 记录 ， 并 限制 返回 一 条 记录 。 

(7) 执行 查询 并 获取 返回 记录 : 


if (Sr = mysql_query ($query, $dbc)) { 
Srow = mysql_fetch array ($r); 


(8) 打印 名 人 名 言 : 

print "<div><pblockquote>{$row['quote']}</blockaquote>- {$row['source']} ， 
这 个 代码 同 view_quotes .php 脚 本 中 的 一 样 ， 但 只 需要 使 用 一 次 。 

(9) 如 果 名 言 是 受 欢 迎 的 ， 就 会 在 显示 标识 之 后 结束 <div> 标 签 : 


if ($row[l'favorite'] == 1) { 
print ' <strong>Favorite! </strong>'; 
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} 

Brint, T/AiV>" 

这 个 条 件 语句 与 delete_quote.php 和 view_quotes .php 脚 本 中 的 一 档 
(10) 如 果 用 户 是 管理 员 ， 创建 用 于 编辑 和 删除 这 条 记录 的 链接 : 


if (is administrator()) { 
print "<p><b>Quote Admin:</b> <a href=\"edit_ quote.php? 
id={$row['quote_id']}\">Edit </a> - + | +- 
<a href=\"delete quote.php? id={$row['quote id']}\"> Delete</a> 
/BSN 
} 


如 果 用 户 是 管理 员 ， 这 个 页 面 就 会 添加 连接 到 编辑 和 删除 脚本 的 链接 。 这 些 链接 的 值 与 
view_quotes .php 脚 本 中 一 样 。 
(11) 如 果 查 询 不 能 运行 ， 打 印 错误 信息 : 


) else { // 没有 运行 查询 。 
print '<p class="error">Could not retrieve the data 
because:<br/>' . mysql_ error ($dbc) . '.</p><p>The query 
"being run was: ' . $query .'</p>'; 


}】 // 结束 查询 条 件 语 向。 

以 上 代码 是 出 于 调试 目的 而 设计 的 。 你 不 应 该 向 公众 暴露 MySQL 错 误 信息 或 引起 错误 的 查 
询 信息 。 

(12) 关闭 数据 库 连 接 : 

mysql_close(s$sdbc); 

之 后 不 需 ;要 再 使 用 数据 库 连 接 了 ， 因此 在 这 里 关闭 它 。 

(13) 创建 连接 到 其 他 页 面 的 链接 : 


print '<p><a href="index.php"> Latest</a> <-> <a href= 
—"index.php?random=true"> Random</a> <-> <a href= 
>"index.php?favorite=true"> Favorite</a><p>'; 


添加 3 个 面向 公众 的 链接 , 每 个 链接 都 指向 这 个 脚本 。 第 1 个 链接 不 会 通过 URL 传 送 任何 值 ， 
它 会 一 直 显 示 最 近 添 加 的 名 言 。 第 ?2 个 链接 ， 通过 URL 传 递 一 个 随机 值 ， 这 会 触发 第 (4) 步 的 查 
询 。 第 3 个 链接 ， 通 过 URL 传 递 受 欢迎 值 ， 这 会 触发 第 (5) 步 的 查询 ， 获 取 一 个 随机 的 受 欢 迎 记 
录 。 

(14) 包含 页 脚 文件 并 完成 页 面 


include('templates/footer.html'); 


3 





o 





























(15) 保存 文件 并 在 Web 浏 览 器 中 测试 (参见 图 13-23)。 
w 提示 


口 通常 主页 是 首先 要 编写 的 脚本 ， 而 不 是 在 最 后 。 但 在 这 个 例子 中 ， 我 想 展示 这 个 页 面 是 


如 何 一 步 步 创建 起 来 的 。 Ee 











378 0130 000000000 


| 全 http:j/10,0,1.,2:8888jindex.php > 





My Site of Quotes 


Latest Random Favorite 




















图 13-23 ”最 近 添 加 的 名 人 名 言 (注意 URL) 





13.12 ”回顾 和 实践 


如 果 你 对 本 节 中 的 内 容 存 有 疑问 ， 可 以 在 本 书 的 论坛 上 提问 ， 网 站 地 址 : www.LarryUllman. 
comyforumy/ 。 





13.12.1 回顾 


口 如 何 调 用 is_agdministrator() 函数 检查 有 不 同 值 的 同一 个 cookie (名 字 为 Samuel) ? 
如 何 检查 有 不 同 值 的 不 同 cookie (名 字 不 为 Samuel) ? 

口 为 什么 在 头 文件 中 引用 样式 表 时 使 用 的 是 css/style.css 而 不 是 ../css/style.css? 
还 可 以 用 什么 方式 引用 样式 表 ? 

口 为 什么 login .php 脚 本 以 那 种 方式 构建 ? 如 何以 更 加 线性 的 方式 来 组 织 这 个 脚本 。 

口 这 个 网 站 还 可 以 使 用 哪些 用 户 自 定义 函数 ? 提示 : 关键 在 于 找到 重复 使 用 的 代码 。 














13.12.2 ”实践 


口 将 登录 表单 修改 成 粘性 表单 。 

口 定义 登录 验证 信息 (cookie 名 和 值 )， 将 它 作 为 配置 文件 中 的 常量 。 接 下 来 在 每 个 页 面 中 
包含 配置 文件 并 使 用 这 些 常量 创建 、 删 除 和 确认 cookie 的 值 。 

口 限制 cookie 的 过 期 时 间 为 15 分 钟 , 接 下 来 在 适当 的 情况 下 ( 即 如 果 cookie 存 在 ), 在 每 个 页 
面 上 再 次 发 送 cookie。 

口 使 用 Session 代替 cookie 。 

口 将 add_quote.php 和 edit_quote.php 表 单 修改 成 粘性 表单 。 

口 修改 view_quotes .php 脚 本 ， 让 管理 员 可 以 以 不 同 的 顺序 显示 名 言 。 提 示 : 创建 类 似 
index.php 脚 本 中 返回 页 面 的 链接 ， 然 后 相应 修改 查询 语句 。 

口 在 将 这 个 网 站 放 到 实际 服务 器 上 之 前 (你 应 该 做 这 件 事 )， 更 新 所 有 代码 ， 以 确保 非 管 理 
员 不 会 看 到 任何 MySQL 错 误 。 


























安装 和 配置 











运行 本 书 示 例 , 需要 3 种 技术 (工具 ): MySQL (数据 库 应 用 程序 )、PHP (脚本 语言 ) 和 Web 
服务 器 应 用 程序 (使 PHP 得 以 运行 )。 附录 A 将 介绍 如 何在 Windows 和 Macintosh 这 两 个 不 同 的 平台 
上 安装 这 些 工具 。 如 果 读 者 使 用 的 是 Web 站 点 主机 ， 它 可 能 已 经 提供 了 这 些 工 具 , 但 由 于 这 些 产 
品 都 是 免费 的 并 且 易 于 安装 ， 因 此 也 可 以 将 它们 安装 在 自己 的 计算 机 上 。 

在 介绍 了 安装 之 后 ， 本 附录 还 将 介绍 使 用 MySQL 和 配置 PHP 的 基础 知识 。PHP 和 MySQL 的 
手册 都 相当 详细 地 介绍 了 它们 的 安装 和 配置 方法 。 如 果 遇 到 问题 ,可 以 详细 地 阅读 这 些 手 册 以 寻 
求解 决 方法 。 


A.1 在 Windows 上 安装 


虽然 你 可 以 在 Windows 系 统 上 分 别 安装 Web 服 务 器 〈 例 如 ，Apache、Abyss 或 IS ) 、PHP 和 
MySQL， 但 是 我 强烈 建议 你 使 用 多 合 一 安装 包 。 这 样 做 更 容易 ， 也 不 太 会 出 现 问 题 。 

用 于 Windows 的 集成 安装 程序 有 很 多 。 XAMPP (www.apachefriends.org ) 和 WAMP 
(www.wampserver.com) 是 最 经 常 提 到 的 两 种 集成 安装 程序 。 在 附录 人 A 中 ， 本 书 将 使 用 XAMPP， 
它 可 以 运行 在 Windows 2000、Windows 2003、Windows XP 和 Vista 上 。 (XAMPP 网 站 上 并 没有 提 
及 Windows 7， 但 它 可 以 运行 在 Windows7 上 。) 

除了 Apache、PHP 和 MySQL 之 外 ，XAMPP 还 安装 了 : 

口 PEAR 一 一 一 个 PHP 代 码 库 ; 

D phppMyAdmin 一 一 有 Web 界 面 的 MySQL 服 务 器 ， 
口 一 个 邮件 服务 器 (用 于 发 送 邮 件 )，; 

口 大 量 有 用 的 扩展 。 

本 书 在 编写 时 ，XAMPP (版 本 1.7.3) 会 同时 安装 PHP 5.31、MySQL 5.1.41、Apache 2.2.14 和 
phpMyAdmin 3.2.4。 

下 面 本 书 将 逐步 介绍 安装 过 程 。 请 注意 ， 如 果 遇 到 任何 问题 ， 都 可 以 访问 本 书 的 支持 论坛 
(www.LarryUllman.comy/forum/) , 但 向 XAMPP 站 点 求助 可 能 会 更 好 一 些 (毕竟 这 是 他 们 的 产品 ) 。 
另外 , 安装 程序 可 以 工作 得 很 好 而 且 使 用 也 很 简单 ， 所 以 本 书 没有 详细 地 介绍 安装 过 程 中 的 每 一 
个 步骤 ， 而 是 着 重 于 更 重要 的 需要 考虑 的 事情 。 
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防 火 墙 

D00000000000000U000UUUUUU LD Windowsd XPServicePack20] 
oo 
DUDXAMPPN UD Apachel] MySQLUUOOOUOUUOUOUOUOOOUO0O0ODODO 
doo 

U0UO0U0D0XAMFEUUOUOUOOUUOUOOOUU0O0O0D0D0Apachel MySQLUDD 
UUOUUOO0O0O000 UnobloccUOu0oD0OU0O0O0000000000D0D0 Windows XP 
D000 DO0O00” 关 DOOOoo2“ DOoooooooo000008s@ 四 ApacheDUU3306] MySQID 
U 23 Mercury UUOUOUODOUOUOOU0OOOO0O0OUOOUUUOU0OU0UO0OUOU0U00D0 
四 加 加 

D0000000000000U windowsUUUUOUOUOO0O0O00D0 Windows 0 0 
本 ET 


瀛 在 Windows 上 安装 XAMPP 


(1) 从 www.apachefriends.org 下 载 Windows 版 XAMPP 的 最 新 版 本 。 

需要 进行 几 次 单 击 才能 找到 下 载 区 ， 最 终 你 会 到 达 一 个 类 似 于 图 A-1 那 样 的 区 域 。 然 后 单 击 
EXE， 这 正 是 我 们 需要 的 东西 。 

(2) 在 计算 机 上 ， 双 击 下 载 的 文件 以 启动 安装 过 程 。 

(3) 出 现 提 示 信 息 后 (图 A-2)， Te in ell nit 

由 于 在 Windows Vista 上 会 遇 到 权限 问题 ， 因 此 不 能 将 其 安装 到 Program Files 目 录 中 。 本 书 建 
议 将 其 安装 到 根 目 录 (如 c:\) 中 。 

无 论 决定 将 程序 安装 到 哪里 ， 都 要 注意 其 安装 位 置 ， 在 本 附录 中 会 多 次 提 到 这 一 点 。 




















XAMPP for Windows 1.7.3, 2009/12/23 三 xAMPP for Windows 


Version Size Content 





XAMPP Windows 1.7.3 Apar hs 2， 3 GD able 9 pe BE 村 局 未 ngine, 
[Basic package nel VO 8 arc 





EXE 51 MB Self-extracting RAR archive 
巴 ZIP 100 MB ZIP archive 


XAMPP Windows 1.7.3 
[Upgrade 1.7.2 to 1.7.3] 


7 EXE 45 MB Self-extracting RAR archive 











@ZIP 89 MB ZIP archive 











图 A-1 从 Apache Friends 网 站 获取 最 新 的 Windows 版 本 的 图 A-2 ”选择 XAMPP 的 安装 位 置 
安装 程序 


( 你 还 可 以 在 桌面 或 开始 菜单 中 创建 快捷 方式 (参见 图 A-3)。 
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| C:MWINDOWS'system32\cmd.exe [- 口 | x] 
扫 和 持 挂 持 持 拉 持 持 持 持 挂 挂 持 拉 持 拓 持 持 持 挂 持 村 持 持 持 持 持 挂 持 持 持 持 拓 持 挂 持 挂 持 持 持 持 村 持 挂 挂 对 对 持 持 挂 持 挂 持 村 对 持 持 持 持 挂 村 村 持 拉 持 持 持 扩 持 村 -| 
提 XhMPP 1.7-3 - Setup | 
bh ees ETS 
## Copyright 2099 Carsten Wiedmann (FreeBSD License> -| 
# huthors: Carsten Wiedmann 《caksten_sttgtegmx-de> : 


av Uogelgesang 《kvoB@apachefriends .org> 
失 提 盾 打 提 持 提 提 村 盾 直 拉杆 林村 村 村 提 拉 打 提 持 持 提 拉 村 相持 村 打 打 拉扯 提 拉杆 林 打 村 相间 拉 打 提 拉 持 提 拉 盾 提 持 持 打 打 拉 提 提 拉 村 林村 村 打 提 村 提 打 村 盾 寺 村 相间 站 寺 提 


Should I add shortcuts to the startmnenu/desktop? Cy/n): yu 














图 A-3 ”本 书 建议 使 用 的 XAMPP 选 项 


在 安装 过 程 中 ，XAMPP 会 打开 了 一 个 控制 台 窗 口 ， 这 个 窗口 会 提示 一 些 其 他 选项 。 

(5) 出 现 提示 时 ， 可 以 选择 XAMPP 的 安装 路 径 。 

(6) 出 现 提 示 时 ， 需 要 提供 XAMPP 的 安装 盘 符 。 

(7) 继续 完成 后 面 的 步 又， 阅读 之 后 按 Enter (或 Return) 前 进 。 

随后 ， 安 装 程序 会 提示 安装 成 功 ， 接 下 来 安装 程序 会 要 求 你 设置 PHP 配 置 中 的 时 区 等 内 容 ， 
按照 提示 做 就 行 了 。 

(8) 安装 完成 后 〈 参 见 图 A-4) ， 按 1 键 启动 控制 面板 。 

安装 完成 后 ,在 控制 台 窗 口中 还 会 出 现 儿 个 选项 。 一 个 选项 是 启动 控制 面板 , 你 也 可 以 单 击 
桌面 或 开始 菜单 中 的 快捷 方式 启动 它 ， 但 前 提 是 你 已 经 在 第 (4) 步 创建 了 它们 。 

(9) XAMPP 控 制 面板 可 以 启动 、 停 止 和 配置 XAMPP (参见 图 A-5)。 





























XAMPP Control Panel Application 


[2 AMPP Control Panel Shell 
{Apache Friends Edition) 





























Setup 
Modules Port-Check 
Apache Explore 
Mysa 
Filezilla [star ] Refresh 
二 C:MWINDOWSYsystem32\cmd.exe |- 口 | x| 人 








3 回 
加 





和 





1. start XhMPP Control Panel Tomcat Exit 
2. relocate XhMPP 

3 rnt pabh: Coen> KAMPP Control Panel Version 2.5.8 (2009-07-28) 

4. disable Server Side Includes (SSI> EO 条 的 人 本 久生 和 近 生 全 后 下 2 和 

5- enable IPu4 only (current: IPv4/6 《auto?》 Windows 5.1 Build 2600 Platform 2 Service Pack 3 

6. disable mod_perl Current Directory: C:\xampp 

?7. disable fpache::ASP sh Status Check OR 

x Exit 

3 choose (1-?/x): | 


a 


| 2»| > 














图 A-4 ”安装 完成 图 A-5 如 果 运 行 了 某 种 防火 墙 ， 在 启动 Apache 
或 其 他 应 用 程序 时 可 能 会 看 到 类 似 的 消 
息 。 参 见 框 注 “防火 墙 ” 














(10) 使 用 控制 面板 启动 Apache、MySQL 和 Mercury。 

本 书 中 所 有 的 示例 都 要 启动 Apache。 完成 第 12 章 的 内 容 需 要 启动 MySQL。Mercury 是 XAMPP 
安装 的 邮件 服务 器 。 如 果 要 使 用 PHP 发 送 Email 就 要 启动 它 ( 详 见 第 8 章 )。 

(11) 立即 为 MySQL 用 户 设 置 密 码 。 

本 章 稍 后 将 介绍 如 何 完成 这 一 步 。 
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Vv 提示 
口 XAMPP 的 控制 面板 有 很 多 管理 链接 ， 它 们 可 以 连 到 不 同 的 页 面 上 (你 自己 的 服务 器 上 ) 
或 其 他 资源 上 (参见 图 A-6)。 


3 xAMPP 1.7.3 - Mozilla Firefox 


(< 2 Cay http: i/localhost/xampp} 


XAMPP for Windows S23 
XAMPP 1.7.3! 


Et Congratulations: 
Status You have successfully installed XAMPP on this system! 
Securit 
全 Now you can start using Apache and Co, You should first try >Statuse on 
Components the left navigation to make sure eyerything works fine, 


phpinfoO) For OpenSSL support please use the test certificate with https://127.0.0.1 
pearinfo() or https: /flocalhost 
perlinfo() 人 有 
aspinfol) And very important! Big thanks for help and support to Nemesis, Kris, 
Boppy, Pc-Dummy and all other friends of XAMPP! 
ED ol Good luck, Kay Yogelgesang + Kai 'Oswald' Seidler + Carsten Wiedmann 
Biorhythm 
Instant Art 
Flash Art 
Phone Book 
ADOdb 
Guest Book 


phpMyAdmin 
‘Webalizer 





图 A-6 ”控制 面板 中 链接 的 XAMPP 基 于 Web 的 前 导 页 面 


口 请 参见 本 章 后 面 的 A.3 节 ， 学 习 如 何 通 过 编辑 php . ini 文 件 来 配置 PHP。 
口 Web 根 目录 (放置 PHP 脚 本 以 供 测 试 的 地 方 ) 是 XAMPP 安 装 目录 中 的 htdocs 文 件 夹 。 这 
里 安装 的 目录 是 C:xwampp\htdocs。 


A.2 在 Mac OSX 上 安装 


由 于 那些 现成 的 安装 包 , 在 Mac OS X 上 安装 MySQL 和 PHP 曾 经 (也 许 仍 然 ) 非常 简单 。Mac 
OS X 已 经 使 用 Apache 作 为 其 Web 服 务 器 了 ,而 且 还 安装 了 PHP 的 某 个 版 本 ,只 是 没有 启用 。 感 谢 
Marc Liyanage (www.entropy.ch) ， 他 为 了 支持 Mac 操 作 系 统 做 了 很 多 工作 ， 因 而 更 多 PHP 的 当前 
版 本 和 功能 丰富 的 版 本 都 能 够 很 容易 地 安装 。 

本 书 称 安装 “曾经 ”并 且 “ 也 许 仍然 ”简单 ， 是 因为 这 取决 于 所 使 用 的 硬件 和 操作 系统 。 在 
编写 本 书 时 ， 最 新 的 Mac 使 用 Intel 64 位 处 理 器 。 这 些 计 算 机 可 以 运行 32 位 和 64 位 软件 。 为 Mac OS 
X10.5 (Leopard) 构建 的 Apache 是 64 位 版 本 的 (假设 计算 机 支持 ), 这 意味 着 PHP 以 及 大 量 由 PHP 
使 用 的 库 也 必须 是 64 位 版 本 的 ， 这 并 不 容易 做 到 。 

由 于 在 Mac OS X 10.5 上 使 用 64 位 版 本 的 Apache (如 果 可 以 的 话 ) 会 出 现 兼 容 性 问题 ， 所 以 
本 书 决定 介绍 一 种 更 为 常见 的 、 非 常 简 单 的 方法 ， 并 建议 使 用 MAMP (www.mamp.info) 集成 
安装 程序 。 它 既是 免费 的 又 具有 官方 版 本 ， 易 于 使 用 ， 而 且 不 会 影响 到 在 操作 系统 中 内 置 的 
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Apache。 

除了 Apache、PHP 和 MySQL，MAMP 还 会 安装 phpMyAdmin (基于 Web 界 面 的 MySQL) 以 及 
大 量 有 用 的 PHP 扩 展 。 在 编写 本 书 时 , MAMP (版 本 1.9.4) 安装 了 PHP 5.2.13 和 5.3.2、MySQL 5.1.44 
Apache 2.0.63 以 及 phpMyAdmin 3.2.5。 

下 面 本 书 将 逐步 介绍 安装 过 程 。 请 注意 ， 如 果 遇 到 任何 问题 ， 都 可 以 访问 本 书 的 支持 论坛 ， 
但 向 MAMP 站 点 求助 可 能 会 更 好 一 些 ( 毕 竞 这 是 他 们 的 产品 )。 男 外 ,安装 程序 可 以 工作 得 很 好 
并 且 易 于 使 用 , 所 以 本 书 没有 详细 地 介绍 安装 过 程 中 的 每 一 个 步 又， 而 是 着 重 于 更 重要 的 需要 考 

之 在 Mac OS X 上 安装 MAMP 


(1) 从 www.mamp.info 下 载 MAMP 的 最 新 版 本 。 

在 首页 单 击 Download ， 然 后 单 击 MAMP & MAMP PRO 1.9.4 (参见 图 A-7)。( 随 着 MAMP 新 
版 本 的 发 布 ， 链 接 和 文件 名 也 会 随 之 变化 。) 

两 种 产品 的 下 载 文 件 是 一 样 的 。 实 际 上 ，MAMPPro 只 是 有 一 个 更 好 的 控制 和 自 定义 MAMP 
软件 的 界面 。 

(2) 在 计算 机 上 ， 双 击 下 载 的 文件 ， 安 装 磁盘 镜像 (参见 图 A-8)。 























Download: MAMP & MAMP PRO 1.9.4 
Published: 2010-10-26 


This download package contains the free MAMP and a free 14-day trial of MAMP PRO. 
MAMP can be used stand-alone without MAMP PRO. 


The trial Version of MAMP PRO can be upgraded to the full version by buying a serial number. 
You can order a serial number at our shop. 


All MAMP PRO updates in the current major version (1.x) are free of charge 
If you want to update MAMP PRO from e.g. 1.7.2 to 1.9.4 just use the serial number you already got 
® Requirements: min. Mac OS X 10.4 
(This version of MAMP & MAMP PRO is indeed compatible with Mac OS X 10.6 Snow Leopard.) 
® platform: Universal Binary 
e File type: dmg 
® File size: app. 170 MB 





® Translations MAMP: English, French, German, Italian, Japanese, Russian, Spanish 


® Translations MAMP PRO: English, German, Japanese 
































图 A-7 ”从 www.mamp.info 中 的 这 个 页 面 里 下 载 MAMP 图 A-8 ”下载 的 MAMP 磁 盘 镜 像 中 的 内 容 


(3) 将 MAMP 文 件 夹 从 磁盘 镜像 复制 到 Applications 文 件 夹 。 

如 果 更 喜欢 商业 版 的 MAMP PRO， 请 复制 相应 的 文件 夹 (由 于 它 是 连接 到 MAMP 的 界面 ， 
因此 需要 两 个 文件 夹 ) ， 它 有 一 个 14 天 免费 试用 的 版 本 。 

不 管 选择 的 是 哪个 文件 夹 ， 请 注意 必须 将 其 放置 在 Applications 文 件 夹 中 ,而 不 能 放 在 一 
个 子 文件 夹 或 计算 机 上 的 其 他 目录 中 。 

(4) 打开 Applications/MAMP (或 ApplicationMMAMP PRO) 文件 夹 。 

(5) 双击 MAMP (或 MAMP PRO) 应 用 程序 ， 启 动 程序 (参见 图 A-9)。 

需要 花 一 小 会 儿 时 间 来 启动 服务 器 ， 然 后 就 会 看 到 如 图 A-9 所 示 的 结果 (MAMP) 或 参见 图 
A-10 所 示 的 结果 (MAMP PRO ) 。 
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图 A-9 MAMP 应 


用 程序 ， 用 于 控制 和 


配置 Apache、PHP 和 MySQL 
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图 A-10 MAMP PRO 应 用 程序 , 用 于 控制 和 配 
置 Apache、PHP 和 MySQL 等 


当 启 动 MAMP 时 ， 默 认 Web 浏 览 器 会 打开 起 始 页 (参见 图 A-11)。 通 过 这 个 页 面 可 以 浏览 到 
正在 运行 的 PHP 版 本 以 及 它 的 配置 ， 还 有 使 用 了 phpMyAdmin 的 MySQL 数 据 库 界 面 。 
如 果 是 MAMP PRO， 你 可 以 单 击 WebStart 按 钮 访问 这 个 界面 (参见 图 A-10)。 








enNno 


MAMP = 











Q. Welcome to MAMP 


"| If you can see this page, MAMP is installed 
on your Mac and everything is working! 


PHP 


Te see rhe PHP conflgeration, you can watch the output of ohplnto 


My5QL 


The MySQL Database can be admmnitrated with phpMyAdmin. 


To connect to the MySQL Server from your own scripts use the fogowing connection 


paramerery 
Host localhost 

Po 8859 

User root 

Password root 

Eampie- 

Slink = sysql_cornect( "localhost' , ‘root", ‘root"); 


OF YOu can Connect using an UNIX Socket 


Socket lApplicatons/ MAMP /tmp/ mys ol mysql. soc 
User root 

Password toot 

Eampie: 


Sink ~ mysql.cornect( 
';/Applicotions/ /MANP/tmp/mysql/mysql ,50cK" , 
root 
00t 

六 





MAMP 到 MAMP PRO 1.2.1 released 

MAMP & MAMP PRO 1.9 released with PHP 5.3) 
News 

Firs! Moments wh MAMP 

There is a great screencass by Chris Coyier on "Pest 
Momeets with MAMFP- Chack 民 ousl 

HowTe. Creste a local environment vying MAMP 


The team of drupal.org wrote 2 great step by step 
wrorial Create a local environment vsing MAMF”, Many 
thanksl Check it outl 


New Online Documentation for MAMP & MAMP PRO 
We have just finished the transition of the MAMP PRD 
POF -based documensation exo an online version. Some 
eraries have been vpdated cehers have been added We 
Aso started to 26d a documentation for the free MAMP, 
but wt is wery basic ye 


MAMP and IPYS 


We are getting lots of questions regarding MAMP / 
MAMP FRO and IPYG. Doe5 It work? Yes it does! 











图 A-11 


MAMP 起 始 页 
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(6) 要 启动 、 停 止 和 配置 MAMP， 请 使 用 MAMP 或 MAMP PRO 应 用 程序 (参见 图 A-9 和 图 
A-10), 

关于 应 用 程序 本 身 没 有 什么 需要 多 说 的 (这 是 一 个 很 好 的 程序 ), 但 如 果 单 击 Perferences， 则 
可 以 调整 应 用 程序 的 行为 (参见 图 A-12)， 从 而 设置 要 运行 的 PHP 版 本 ， 等 等 。 

MAMPPRO 也 可 以 创建 不 同 的 0 口 口 口 ( 即 不 同 的 网 站 ), 可 以 调整 Apache 配 置 和 运行 方式 ， 
可 以 使 用 动态 的 DNS 以 及 修改 Email 的 发 送 方式 ， 等 等 。 

(7) 立即 为 MySQL 根 用 户 设置 密码 。 

本 章 稍 后 将 介绍 如 何 进行 设置 。 

ww 提示 
口 就 我 个 人 而 言 ， 我 非常 喜欢 使 用 MAMP ， 因 为 它 是 免费 的 。 我 也 不 喜欢 花 钱 的 东西 ， 但 
我 发 现 花 点 钱 买 一 个 MAMP PRO 还 是 很 划算 的 。 
口 参见 本 章 最 后 的 A.3 节 ， 学 习 如 何 通过 编辑 php . ini 文 件 来 配置 PHP。 
口 你 可 能 希望 将 Apache Document Root (参见 图 A-13) 设置 为 主 文件 夹 中 的 Sites 上 目录。 这 
样 做 之 后 ， 请 确保 备份 Web 文 档 和 其 他 文件 。( 此 时 正在 执行 常规 备份 ， 对 吗 ? ) 
口 MAMP 还 提供 了 一 个 Dashboard 小 工具 ， 用 于 管理 Apache 和 MySQL 服务 器 。 
口 Web 根 目录 (用 于 放置 PHP 脚 本 以 进行 测试 的 地 方 ) 是 MAMP 安 装 目录 下 的 htdocs 文 件 

夹 。 这 里 安装 的 目录 是 Applications/MAMP/htdocs。 


























So MAMP So MAMP 











| start/Stop Ports PHP Apache | Start/Stop Ports PHP Apache | 








Document Root 





町 Stop Servers when quitting MAMP 


| Start Servers when starting MAMP 
加 Check for MAMP PRO when starting MAMP 


/Users/larryullman/Sites| 


pa 







































































忆 | Open start page at startup Select 

Start page url 

/IMAMP/ 

(一 Cancel ) lO ( Cancel ) 人 Gok 3 
ld Preterences. p De Preferences.. pa 
E Quit 4 ( Quit 2 
图 A-12 ”这 5 个 选项 决定 着 当 启动 和 停止 图 A-13 ”MAMP 人 允许 修 改 Web 文 档 放置 的 位 置 
MAMP 应 用 程序 时 会 发 生 什么 
A.3 PHP 配置 


自己 安装 PHP 的 一 个 好 处 在 于 ， 你 可 以 按照 自己 的 喜好 来 进行 配置 。 PHP 如 何 运 行 可 由 
php.ini 文 件 决定 ， 该 文件 通常 在 PHP 安 装 时 创建 。 
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需要 考虑 调整 的 两 个 重要 配置 是 display_errors 和 error_reporting ( 均 在 第 3 章 中 讨论 
过 )。 要 修改 任何 设置 ， 请 打开 PHP 配 置 文 件 ， 按 照 需要 对 其 进行 编辑 ， 然 后 保存 该 文件 ， 并 重 
启 Web 服 务 器 。 


更 改 PHP 配 置 


(1) 在 Web 浏 览 器 中 执行 调用 phpinfo() 函数 的 脚本 (参见 图 A-14)。 
phpinfo () 国 数 在 第 1 章 介绍 过 ， 它 可 以 显示 PHP 安 装 信息 














(< = 奈 开 dy | 加 http:Wlocalhostjxamppjphpinfo.php -| | 是- £ 


PHP Version 5.3.1 php 











System Windows NT YYINMBP 5.1 build 2600 (Windows XP Professional Service Pack 
3)i586 

Build Date Nov 20 2009 17:20:57 

Compiler MSVC6 (Visual C++ 6.0) 

Architecture x86 

Configure Command cscript inologo configure.js "--enable-snapshot-build" 

Server API Apache 2.0 Handler 





Virtual Directory Support enabled 
Configuration File (php.in) | no vaiue 
ath 


Loaded Configuration File |C:xampp\php\iphp.ini 
Scan this dir for additional |inone) 




















‘inifiles 

Additional .inifiles parsed |inone) 

PHP API 20090626 

PHP Extension 20090626 

Zend Extension 220090626 

zend Extension Build API220090626,TS,YC6 v 
< > 








图 A-14 ”调用 phpinfo() 函数 的 部 分 输出 


(2) 在 浏览 器 输出 中 ， 查 找 加 载 的 配置 文件 。 

这 个 文件 后 面 的 值 就 是 配置 文件 的 地 址 。 地 址 可 能 像 这 样 : C: \xampp\php\php.ini 或 
/Library/ApplicationSupport/appsolute/MAMP PRO/conf/php.ini, 

如 果 配 置 文件 后 面 没有 值 ， 就 说 明 你 的 服务 器 上 没有 php .ini 文 件 。 这 种 情况 下 ,你 需要 从 
www.php.net 上 下 载 PHP 源 代码 ， 从 中 找到 这 个 文件 。 

(3) 在 任意 文本 编辑 器 中 打开 php .ini 文 件 。 

(4) 按照 需要 修改 设置 。 

根据 你 使 用 的 操作 系统 ， 你 可 能 需要 管理 员 身 份 或 输入 密码 才 可 以 修改 该 文件 。 

该 文件 包含 很 多 指令 。 通 过 在 前 面 加 分 号 ， 可 以 将 某 些 行 注释 掉 (使 其 无 效 )。 

(5) 保存 php .ini 文 件 。 

(6) 重启 Web 服 务 器 。 
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无 需 重启 整个 计算 机 ， 只 需 重启 Web 服 务 器 (如 Apache) 即 可 。 

V 提示 

口 还 可 以 使 用 phpinfo () 脚本 来 检查 变更 的 配置 是 否 起 作用 了 。 

口 如 果 编 辑 了 php . ini 文件 并 重启 了 Web 服 务 器 ,而 这 些 修改 没有 起 作用 ,请 确认 编辑 了 适 

当 的 php . ini 文件 〈 所 用 的 计算 机 上 可 能 有 多 个 该 文件 )。 

口 在 Mac OS X 上 的 MAMP PRO 会 为 php .ini 维 护 一 个 模板 ， 必 须 在 MAMP PRO 中 编辑 
php ,ini 文件。 在 使 用 MAMP PRO 时 ， 如 果 修 改 PHP 配 置 ， 可 以 单 击 File >Edit Template > 
PHP X.X.X php.ini。 





























启用 邮件 


U0O0UD0UPHPUOUOUOOO0OO0ODU sendmaiiUOOOOOUOOU0O000mailo0U0O00 
UUUmailOUU0U0OU0U000Upne.iniUOUDUsnce0000 windowsUU0O00D0 
IntemetlUUOUOUSMIEUOUOUUOOOUOUOUOUOUOOUOISPU SMIPIUUUOUD 
国 

UDU WindowsUUUO0O0OU0OSMIPEUUOOU0UD MercurxyUUDOUXAMPPUUOUOOODOD 
U000D0XAMFEUUUDOOUOO0OO0OD 

MacOsxUUUUUOUU0UD0O0D0 postxU sendmaillUDOUOUOOOOOUO0OD0OU Soogle 
UUUU00Macosx00u000000000000 

UUOU0O00OUPHPUOUOOOUUOOUOOUOU0O0O0OD0OSMTPUODOD 


A.4 _ MySQL 界面 


在 第 12 章 和 第 13 章 , 我 们 使 用 PHP 脚 本 与 MySQL 进 行 交 互 。 如 第 12 章 所 述 ,最 好 的 调试 方式 
是 使 用 其 他 应 用 程序 来 执行 SQL 命令 。 了 解 如 何 使 用 独立 的 MySQL 界 面 非 常 重要 。 这 里 会 介绍 两 
种 最 常见 的 界面 。 
A.4.1 使 用 MySQL 客 户 端 

MySQL 软 件 包 含 了 一 个 重要 的 工具 ， 称 作 MySQL 客 户 端 。 该 应 用 程序 提供 了 与 MyYSQL 服 务 
器 进行 通信 的 界面 。 这 是 一 个 命令 行 工具 ， 它 在 Linux 和 Mac OS X 上 必须 使 用 Terminal 应 用 程序 
访问 ， 而 在 Windows 上 则 必须 使 用 命令 提示 符 (DOS ) 。 

过 使 用 MySQL 客 户 端 

(1) 确保 MySQL 服 务 器 正在 运行 。 

(2) 找到 MySQL 的 bin 目录 。 

要 连接 到 客户 端 , 需要 知道 它 的 位 置 。MySQL 客 户 端 可 以 在 安装 目录 的 bin 目 录 中 找到 。 本 
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书 将 使 用 最 常用 et 目录 。 

如 果 是 自己 安装 的 MySQL, 则 客户 端的 位 置 取决 于 软件 的 安装 位 置 , 但 很 可 能 是 c: \mysal\ 
bin\mysql (Windows) 或 /usr/local/mysql/bin/mysql (Mac OS X and Unix ) 。 

如 果 在 Windows 上 使 用 了 XAMPP, 则 可 能 是 C:\xampp\mysql\bin\mysql (假设 将 XAMPP 


安装 到 了 C:\)。 如 果 在 Mac OS X 上 安装 了 MAMP， 则 MySQL 的 目录 是 /Applications/MAMP/ 
Library/bin/mysql,。 


(3) 访问 命令 提示 符 。 

在 Mac OS X 和 Unix 上 , 可 以 运行 Terminal 应 用 程序 。 在 Mac OSX 上 , 可 以 在 Applications/ 
Utilities 文件 夹 中 找到 它 。 

在 Windows 上 ， 单 击 Start 菜 单 ， 选 择 Run， 然 后 输入 cmd 并 按 回 车 键 (图 A-15)。 








= Type the name of a program, folder document, or 
3 Internet resource, and Windows will open it for you, 


Open: | cmd v 








] Cancel Browse,,, 





图 A-15 ”Windows 上 的 命令 提示 符 
(4) 尝试 连接 到 MySQL 客 户 端 。 
要 进行 连接 ， 请 输入 第 (2) 步 中 的 路 径 名 再 加 上 -u username -p。 因 此 ， 命令 行 应 该 是 


c:\mysql\pbin\mysql -u username -p (Windows) 


或 








/usr/local/mysql/bin/mysql -u username -p (Unix 和 Mac OS X) 


泪 


C:\xampp\mysql\bin\mysql -u username -p (Windows) 


让 


/Applications/MAMP/Library/bin/mysql -u username -p (Mac OS X) 

将 username 替 换 为 要 使 用 的 用 户 名 。 如 果 还 没 创建 过 其 他 用 户 ， 则 这 里 可 以 用 root (root 是 
MySQL 超 级 用 户 )。 如 果 还 没有 为 root 用 户 设置 密码 ， 则 可 以 忽略 -p 标 志 。 

(5) 在 提示 符 中 输入 密码 (参见 图 A-16)。 





























oo C:WINDOWSYsystem32\cmd.exe - C:Wamppimysql\binimysql -u root -p 





=:C: onpp mys bin Nnyegl -uu Foot -p 

nter pas 

elcome to one ~ MySQL wh oe Commands end with ; or \g-. 
our MySQL connection id 

erver version: 5.1.41 Source distribution 


LL 





ype ’help;’ or ’\h’ for help. Type ’\c’ to clear the current input statement. 


ysql> - 








图 A-16 在 Windows 上 成 功 地 访问 了 MySQL 客 户 端 
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这 里 所 必需 的 密码 是 再 连接 时 针对 用 户 名 设置 的 MySQL 密 码 。 只 有 在 第 (4) 步 中 使 用 了 -Pp 选 
项 才 会 看 到 这 个 提示 信息 。 

如 果 在 Mac OS X 上 安装 了 MAMP， 则 root 用 户 的 密码 就 是 root。 如 果 在 Windows 上 安装 
XAMPP， 就 没有 初始 密码 。 

(6) 列 出 可 用 的 数据 库 (参见 图 A-17);: 

SHOW DATABASES; 

SHOW DATABASES 命 令 是 一 个 SQL 查询 ， 列 出 了 寄宿 在 所 安装 的 MySQL 中 并 能 由 当前 连接 的 
用 户 看 到 的 所 有 数据 库 。 


























3 
ht ri ge a 到 








号 
本 
各 
y 
多 
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! information_schema ! 
1 cdcol ! 
上 mysql 

! phpmyadmin 


Imysql> - 


| 
到 | 2| 























图 A-17 在 全 新 安装 MySQL 之 后 ， 只 能 看 到 3 个 默认 数据 库 

(7) 退出 MySQL 客 户 端 。 

键入 exit 或 quit。 

w 提示 

口 如 果 看 到 Can’t connect to local MySQL server through socket... 错 误 消 息 ， 通 常 表示 MySQL 

没有 运行 。 

口 MySQL 客 户 端 是 调试 MySQL 和 PHP 脚 本 的 最 佳 工 具 。 可 以 使 用 MySQL 客 户 端 来 检查 用 户 
权限 ， 并 在 PHP 脚 本 之 外 运行 查询 。 











A.4.2 ”使 用 phpMyAdmin 


phpMyAdmin (www.phpmyadmin.net) 是 一 个 与 MySQL 交 互 的 Web 界 面 ， 它 具有 创建 表 、 导 
入 或 导出 记录 等 功能 。phpMyAdmin 是 一 个 图 形 界面 ， 它 可 能 是 用 PHP 编 写 的 最 流行 的 Web 软 件 ， 
几乎 所 有 的 PHP 托 管 公司 都 提供 这 个 软件 。 实 际 上 ， 多 合 一 安装 包 XAMPP 和 MAMP 也 包含 这 个 
软件 。phpMyAdmin 设 计 合 理 并 且 易 于 使 用 ， 下 面 我 会 介绍 一 些 使 用 技巧 。 

(1) 在 Web 浏 览 器 中 访问 phpMyAdmin (参见 图 A-18)。 
如 果 用 的 是 XAMPP，, 可 以 进入 http://localhost/phpmyadmin/ 访 问 phpMyAdmin, 也 可 以 在 控制 
面板 中 单 击 管理 链接 。 如果 用 的 是 MAMP, 可 以 进入 http:Wlocalhost8888/MAMP/ 访 问 phpMyAdmin。 
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(2) 单 击 左 列 的 数据 库 名 ， 选 择 数 据 库 。 














@ -CG XX 会 [A http:iiflocahostiphpmyadmin} 立 下-| 月 | 
phpMyAdmin 中 Server: localhost 所 
回回 草 Databases _ 蝇 SQL 车 Status 图 Variables_[ 辐 Charsets 鹏 Engines 
剖 Privileges 纺 Processes 总 Export 可 Import 

® cdcol (1) 
® information_schema (28) Actions MySQL 
® mysql 23) 
. en 名 ) MySQL localhost 加 Sewer localhost via TCPHP 
. tesi 

。 加 Server version: 5.1.41 

本 Create new database ® » Protocol version: 10 


Please select a database 





bp User. root@localhost 


























Collation [Loreate | MySQL charset: UTF-8 
到 MySQL connection collation: Unicode (ut) 
utf8_general_ci hd 
ge Web server 
Interface » Apache/2.2.14 (Win32) 
FL DAYW/2 mod_ssl/2.2.14 
i 国 : OpenSSLD.9.8| 
English v mod_autoindex_color 
er 一 PHP#5.3.1 
Theme / Style: | Original mod_apreq2-2009011072.7.1 


» Custom color: ELPeset] mod_perl2.0.4 PerlA5.10.1 


ee 一 ， MYSQL client version: 
Font size: [82% WW 5.1.41 
b PHP extension: mysqli 
phpMyAdmin 


pb Version information: 3.24 总 











图 A-18 ” phpMyAdmin 的 主页 


(3) 在 左 列 中 单 击 一 个 表 名 ， 选 择 该 表 (参见 图 A-19)。 
这 一 步 不 是 必需 的 ， 但 是 做 这 一 步 可 以 简化 很 多 操作 。 








田口 日 MAMP SS 







|， 





MAMP & MAMP PRO powered by appsolute CmbH 











entries 








phpMuA dm 加 localhost myblog 


围 Browse ”办 Structure 入 SQL 万 Search 3:Insert 频 Export_ 圈 Impor 多 Operations 园 Empty 加 Drop 


























回 圆 回回 | Field Type Collation Attributes Null Default Extra Action 
Database 口 entry id int(10) UNSIGNED No None auto_increment We Es 回 
Lmyblog () 阅 口 tme varchar(100) Iatin1_swedish_ci No ”None 5 des | 
口 entry text latin1_swedish_ci No None 上 太 X 轩 器 
gd 口 date_entered datetime No None 卢 X 国 
lenis tt Check AIl/ Uncheck AIl With selected: Zp Xx 图 





学 Print view 蝎 Propose table structure 
了 邓 Add 1 field(s) @ At End of Table O 〇 At Beginning of Table O After Lentry_id el Go 


+ Details... 

















图 A-19 在 左 列 中 选择 数据 库 或 表 ， 在 页 面 右边 可 以 修改 选项 
(4) 使 用 页 面 右边 的 标签 和 链接 可 以 执行 一 些 常 见 任务 。 
一 般 而 言 ， 标 签 和 链接 是 一 些 常用 SQL 命令 的 快捷 方式 。 例 如 ，Browse 标 签 执行 SELECT 查 
询 ， 而 Insert 标 签 会 创建 一 个 表单 ， 用 于 添加 记录 。 
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(5) 使 用 SQL 标签 执行 任意 SQL 命令 。 
你 也 可 以 使 用 SQL Query Window， 它 的 链接 在 数据 库 和 表 名 列表 的 上 面 。 无 论 使 用 哪 种 界 
面 ， 你 都 可 以 测试 PHP 中 使 用 的 查询 语句 ， 这 样 就 无 需 增加 脚本 本 身 的 复杂 度 了 。 





w 提示 
口 还 有 很 多 与 MYSQL 交互 的 客户 端 软件 ， 但 是 MySQL 命 令 行 客户 端 和 phpMyAdmin 是 最 常 
用 的 两 个 软件 。 


A.5 管理 MySQL 用 户 


成 功 安装 了 MyYSQL 之 后 ， 你 就 可 以 创建 用 户 了 。MySQL 用户 是 一 个 基本 的 安全 概念 ， 可 以 
限制 访问 、 影 响 数据 存储 。 需 要 说明 的 是 ,数据 库 应 该 有 多 个 用 户 ， 就 像 操 作 系统 一 样 。MySQL 
用 户 不 同 于 操作 系统 用 户 。 在 你 自己 的 计算 机 上 学 习 PHP 和 MySQL 时 ， 你 不 需要 创建 新 的 用 户 ， 
但 是 你 需要 在 真正 的 网 站 服务 器 上 创建 MySQL 用 户 ， 并 赋予 合适 的 权限 。 

最 初 安装 的 MySQL 只 有 一 个 用 户 (名 为 root) 并 且 没 有 设置 密码 (除非 使 用 MAMP， 它 会 设 
置 一 个 默认 密码 root) 。 在 安装 完成 后 ， 你 应 该 为 该 用 户 创建 一 个 新 密码 。 

在 这 之 后 ， 应 该 创建 其 他 具备 有 限 权 限 的 用 户 。 作 为 规则 ， 不 应 该 使 用 root 用 户 来 完成 常规 
的 日 常 操作 。 


A.5.1 为 root 用 户 设 置 密码 


刚 安装 好 MySQL 时 ， 没 有 建立 任何 有 价值 或 安全 的 密码 。 这 显然 是 一 个 安全 风险 ， 并 且 应 
该 在 使 用 服务 器 之 前 修正 (因为 root 用 户 具 有 最 高 权限 ) 。 

可 以 使 用 phpMyAdmin 或 MySQL 客 户 端 修改 用 户 的 密码 ， 但 前 提 是 必须 运行 MySQL 服 务 器 。 
如 果 MySQL 当 前 没有 运行 ， 请 使 用 本 附录 之 前 所 介绍 的 步骤 启动 它 。 

其 次 ， 如 果 要 修改 root 用 户 的 密码 ， 必 须 以 root 身 份 登录 数据 库 。 






































过 使 用 MySQL 客 户 端 为 root 用 户 设置 密码 








(1) 连接 到 MySQL 客 户 端 。 

请 参见 之 前 介绍 过 的 详细 步骤 。 

(2) 输入 下 面 的 命令 ， 将 thepassword 替 换 为 想 使 用 的 密码 (参见 图 A-20): 

SET PASSWORD FOR 'root'@'localhost' = PASSWORD('thepassword '); 

记 住 , MySQL 的 密码 是 区 分 大 小 写 的 , 所 以 Kazan 和 kazan 是 不 同 的 。 在 实际 引用 密码 之 前 的 
PASSWORD 术 语 告诉 MySQL 要 对 这 个 字符 串 进行 加 密 。 并 且 在 PASSWORD 和 左 括号 之 间 不 能 
任何 空白 字符 。 

G3) 退出 MySQL: 


exit 
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a C:WINDOWSYsystem32\cmd.exe - C:ixamppimysql\binumysql -u root -p - 口 | x| 
ysql> SET PASSWORD FOR ’root’@’localhost’ = PASSWORDC’thepassword’»; | 
uery OK. @ rows affected 〈《B.99 sec)> | 
ysql> 司 





4 



































图 A-20 ”安装 完 软件 之 后 ， 应 该 立即 为 root 用 户 建立 安全 的 密码 
(4) 再 次 登录 MySQL 客 户 端 ， 测 试 新 密码 。 


既然 密码 已 经 建立 起 来 了 , 就 需要 向 连接 命令 中 添加 -p 标 志 。 将 会 看 到 Enter password: 提 示 ， 
在 这 里 输入 刚刚 创建 的 密码 即 可 。 














过 使 用 phpMyAdmin 为 root 用 户 设置 密码 








(GD 在 Web 浏 览 嚣 中 打开 phpMyAdmin。 

参见 前 面 介绍 的 步骤 了 解 详细 信息 。 

(2) 在 主页 中 ， 单 击 Privileges 标 签 。 

任何 时 候 你 都 可 以 单 击 左上 角 的 主页 图 标 回 到 主页 。 

(3) 在 用 户 列表 中 ， 在 root 用 户 行 中 单 击 Edit Privileges 图 标 (参见 图 A-21) 。 

















硬 Databases :六 SQL 故 Status 必 





Variables [条 Charsets” 胡 Engines 名 Privileges 纺 Processes 





安 User overview 








mB GD EE GL OT 
User Host Password Global privileges 1 Grant Action 
口 root localhost Yes ALL PRIVILEGES Yes 入 
口 usemame localhost Yes SELECT, INSERT, UPDATE, DELETE, EXECUTE No 岛 Edit Privileges 





图 A-21 phpMyAdmin 显 示 的 MySQL 用 户 列表 




















(4) 使 用 结果 页 下 面 的 Change Password 表单 (参见 图 A-22) 修改 密码 。 





-Change password 





OO No Password 
@ Password: 
Password Hashing: ”加 MySQL 4.1+ 

O MySQL 4.0 compatible 





了 
Generate Password Generate ) 








(GoY 
p04 


























图 A-22 phpMyAdmin 中 更 新 MySQL 用 户 密码 的 表单 


(5) 如 果 需 要 ， 在 phpMyAdmin 的 配置 文件 中 修改 root 用 户 的 密码 。 

修改 root 用 户 的 密码 后 ，phpMyYAdmin 一 般 会 阻止 访问 MySQL 服 务 器 。 原 因 是 在 本 地 安装 的 
phpMyAdmin 一 般 会 使 用 root 用 户 连 接 MySQL, 而 root 用 户 的 密码 被 硬 编码 在 了 配置 文件 中 。 在 完 
成 步骤 (~ 步骤 (4 后 ， 在 phpMyAdmin 目录 [比如 ，/Applications/MAMP/bin/PhpMyRAdmin 











1 
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(Mac OS X 下 的 MAMP ) 或 C:\xampp\phpMyaAdmin (Windows 下 的 XAMPP)] 下 找到 config.inc. 
php 文 件 。 在 任意 文本 编辑 器 或 IDE 中 打开 该 文件 ， 在 以 下 代码 中 ， 用 新 密码 替换 旧 密 码 .: 
scfg['Servers'] [$il]l['password'] = 'the new password'; 


保存 文件 ， 在 Web 训 览 器 中 重新 加 载 phpMyAdmin。 
A.5.2 ”创建 用 户 和 权限 


在 成 功 地 安装 和 运行 MySQL， 并 为 root 用 户 建立 了 密码 之 后 ， 就 该 添加 其 他 用 户 了 。 为 了 改 
善 数据 库 的 安全 性 ， 应 该 坚持 创建 新 用 户 来 访问 数据 库 ， 而 不 是 一 直 使 用 root 用 户 。 

MySQL 权 限 系统 用 于 确保 在 特定 的 数据 库 上 授权 特定 的 命令 。 例 如 Web 主 机 可 以 利用 其 确保 
访问 多 个 数据 库 的 多 个 用 户 不 会 互相 干扰 。MySQL 系 统 中 的 每 个 用 户 都 可 以 访问 特定 主机 ( 计 
算 机 ) 上 特定 数据 库 中 的 特定 功能 。root 用 户 (MySQL 的 root 用 户 ， 而 不 是 操作 系统 的 ) 具有 
最 高 的 权限 ， 可 以 用 于 创建 子 用 户 ， 然 而 子 用 户 也 可 以 具备 像 root 用 户 那样 的 权限 (这 是 很 不 
可 取 的 )。 

当 一 个 用 户 尝 试 在 mysql 服 务 器 上 做 些 什 么 时 , MySQL 首先 会 检查 该 用 户 是 否 有 权限 连接 到 
服务 器 (基于 用 户 名 、 用 户 密码 和 MySQL 数 据 库 中 user 表 中 的 信息 )。 然 后 ，MySQL 会 检查 用 
户 是 否 有 权限 在 给 定 的 数据 库 上 运行 特定 的 SQL 语 句 一 一 例如 , 选择 数据 、 插 入 数据 或 创建 新 表 。 
表 A-1 列 出 了 可 以 为 每 个 用 户 设 置 的 权限 。 


表 A-1 MySQL 权限 













































































权 限 允许 的 操作 
SELECT 从 表 中 读 取 行 

INSERT 向 表 中 添加 新 数据 行 

UPDATE 修改 表 中 现 有 数据 

DELETE 从 表 中 删除 现 有 数据 

INDEX 在 表 中 创建 和 删除 索引 

ALTER 修改 表 的 结构 

CREATE 新 建 表 或 数据 库 

DROP 删除 现 有 表 或 数据 库 

RELOAD 重新 加 载 grant 表 (由 此 来 发 布 用 户 变化 ) 
SHUTDOWN 停 目 MySQL 服务 器 

PROCESS 查看 和 停止 现 有 的 MySQL 进 程 
FILE 从 文本 文件 向 表 中 导入 数据 
GRANT 新 建 用户 

REVOKE 移 除 用 户 权 限 











在 MySQL 中 设置 用 户 和 权限 有 很 多 种 方式 ， 但 是 我 们 建议 使 用 MySQL 客 户 端 和 GRANT 命 令 
手动 完成 。 其 语法 类 似 于 : 
GRANT privileges ON database.* TO'username'@'hostname' IDENTIFIED BY 'password'; 


在 该 语句 的 privileges 部 分 ， 可 以 列 出 来 自 表 A-1 中 的 特定 权限 ， 或 者 可 以 通过 使 用 ALL 
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来 指定 所 有 的 权限 (这 是 很 不 明智 的 )。 该 语句 的 database.* 部 分 指出 了 用 户 可 以 使 用 哪个 数 
据 库 或 表 。 可 以 使 用 database. tablename 这 样 的 语法 来 指定 表 的 名 字 , 或 者 使 用 * . * 来 指定 所 
有 的 数据 库 (同样 也 是 很 不 明智 的 )。 最 后 ， 可 以 指定 用 户 名 和 一 个 密码 。 

用 户 名 的 最 大 长 度 是 16 个 字符 。 在 创建 用 户 名 时 ， 请 确保 避免 使 用 空格 (可 以 用 下 划 线 代 
禁 )， 并 注意 用 户 名 是 区 分 大 小 写 的 。 

主机 名 是 允许 用 户 连 接 的 计算 机 。 它 可 以 是 域名 ， 比 如 www.example.com 或 IP 地 址 。 一 般 来 
讲 , localhost 是 限定 的 特定 主机 名 , 也 就 是 说 使 用 运行 有 MySQL 数 据 库 的 计算 机 上 的 MySQL 用 户 
连接 。 要 指定 任意 主机 ， 可 以 在 主机 名 中 使 用 通配符 (%$): 

GRANT privileges ON database.* TO 'username'@ '%' IDENTIFIED BY'password '; 

我 不 推荐 这 种 方式 。 在 创建 用 户 时 ， 最 好 明确 并 限制 权限 。 

密码 没有 长 度 限制 ， 但 也 是 区 分 大 小 写 的 。 在 MySQL 数 据 库 中 密码 是 经 过 加 密 的 ， 这 意味 
着 无 法 找 回 纯 文 本 格式 的 密码 。 省 略 IDENTIFIED BY 'password' 子 句 可 以 不 要 求 用 户 输入 密 
码 (但 这 也 是 应 该 避免 的 ) 。 
在 这 个 过 程 的 示例 中 ， 我 们 将 创建 两 个 新 用 户 ， 他 们 有 具有 在 temp 数 据 库 上 的 特定 权限 。 记 
住 ， 只 能 为 用 户 设置 在 现 有 数据 库 上 的 权限 。 接 下 来 的 步骤 还 将 介绍 如 何 创建 数据 库 。 





























在 phpMyAdmin 中 创建 用 户 

DDOphpMyAdmin[| DD DOO0UDPphpMyAdminl| DD D0 PrvilegesU DD 0 Privileges[] 
DUOUUUAddANew UseUUUO0Oo0o0o0o0o00oouoooo0oo0o00u0so000050 
国 

本 
DUOOOO0O0O0O000Ssoouuuuuuuuuouuuuuuouuuuuuuu000 
国 

DUOUOOUUOO0OO0ODO PrviesgesUUOOOOOOUOOOUOOOOO0OO0ODO0D 


信使 用 GRANT 创建 新 用 户 

(1) 作为 root 用 户 登录 MySQL 客 户 端 。 

这 一 步 已 经 介绍 过 了 。 必 须 使 用 能 够 创建 数据 库 和 其 他 用 户 的 用 户 登 录 。 
(2) 创建 一 个 新 数据 库 (参见 图 A-23): 


CREATE DATABASE temp; 

















BOO PHP: Visual QuickStart Guide 

Welcome to the MySQL monitor. Commands end with ; or \g. 四 
Your MySQL connection id is 28 

Server version: 5.1.44 Source distribution 





Type ‘help;' or '\h' for help. Type '\c' to clear the current input 
statement. 

mysql> CREATE DATABASE temp; 

Query OK, 1 row affected (9.09 sec) 


mysql> 目 

















图 A-23 ”创建 新 数据 库 
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如 果 所 用 的 MYSQL 服务 器 上 还 没有 temp 数 据 库 ， 那 么 就 使 用 CREATE DATABASE temp 创 建 
一 个 (后跟 一 个 分 号 ， 这 是 MySQL 客 户 端 所 必需 的 )。 

(3) 在 temp 数 据 库 中 创建 一 个 具有 管理 员 级 别 权限 的 用 户 (参见 图 A-24): 
INDEX 


GRANT SELECT, INSERT, UPDATE, DELETE, CREATE, DROP, ALTER, 
”ON temp.* TO 'llama'@ 'localhost' IDENTIFIED BY 'camel'; 














加 站 站 PHP: Visual QuickStart Guide 

mysql> GRANT SELECT, INSERT, UPDATE, DELETE, CREATE, DROP, ALTER, | 
INDEX ON temp.* TO 'llama'@'localhost' IDENTIFIED BY “camelL ' ; 
Query OK, @ rows affected (0.080 sec) 








mysqt> 目 








图 A-24 ”为 单个 数据 库 创建 管理 员 级 别 的 用 户 

用 户 11ama 可 以 在 temp 数 据 库 中 进行 创建 表 、 修 改 表 、 插 入 数据 、 更 新 数据 等 操作 。 这 基本 
上 包括 了 除 创建 新 用 户 之 外 的 所 有 管理 员 级 别 的 权限 。 请 确保 设置 了 密码 一 一 也 许 应 该 使 用 一 个 
比 这 里 更 复杂 的 密码 一 一 另外 ， 最 好 指定 一 个 特定 的 主机 。 

(4) 创建 一 个 对 数据 库 只 有 基本 访问 权限 的 用 户 (参见 图 A-25): 


GRANT SELECT, INSERT, UPDATE, DELETE ON temp.* TO 'webuser'@ 
IDENTIFIED BY 'BroWslng'; 

















'localhost' 














人 OA PHP: Visual QuickStart Guide 
mysql> GRANT SELECT, INSERT, UPDATE, DELETE ON temp.* TO 'webuser' | 国 


@'localhost' IDENTIFIED BY ‘BroWslng'; 
Query OK, © rows affected (0.080 sec) 





mysql> 目 


A-25 ”对 于 同一 个 数据 库 ， 该 用 户 具 备 更 受 限制 的 权限 














| 

















现在 ， 普 通 的 wepuser 用 户 可 以 浏览 记录 (在 表 中 SELECT)， 以 及 添加 、 编 辑 和 删除 记录 ， 
但 该 用 户 不 能 修改 数据 库 的 结构 。 当 建立 起 用 户 和 权限 之 后 ,应 该 采用 自 底 向 上 的 工作 方式 , 在 
任何 时 候 都 启用 尽 可 能 少 的 访问 权限 。 

(5) 应 用 更 改 (参见 图 A-26): 


FLUSH PRIVILEGES; 











@ 1 pHP: Visual QuickStart Guide 


mysql> FLUSH PRIVILEGES; 
Query OK, © rows affected (0.00 sec) 





mysql> 目 























图 A-26 ”在 尝试 用 新 创建 的 用 户 访问 MySQL 之 前 ， 请 不 要 忘 了 这 一 步 











在 告诉 MySQL 重 置 其 可 接受 的 用 户 和 权限 列表 之 前 ， 刚 刚 做 出 的 修改 不 会 产生 影响 ， 这 也 
是 这 个 命令 要 做 的 事情 。 忘 记 这 一 步 会 导致 新 创建 的 用 户 无 法 访问 数据 库 , 这 是 一 种 常见 的 错误 。 


有 os 
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Vv 提示 

口 任何 名 字 以 test_ 开 头 的 数据 库 都 能 够 被 有 权限 连接 到 MySQL 的 用 户 修 改 。 因 此 ， 要 小 
心 不 要 用 这 种 方式 为 数据 库 命名 ， 除 非 它 真 的 是 用 于 实验 的 。 

口 REVOKE 命 令 用 于 移 除 用 户 和 权限 。 











深入 学 习 PHP 的 资产 








本 书面 向 初级 PHP 程 序 员 ， 为 他 们 的 学 习 提 供 了 一 个 良好 的 基础 。 由 于 更 加 专注 于 基础 ， 因 
此 本 书 忽略 或 跳 过 了 一 些 主题 。 本 附录 列 出 了 大 量 有 用 的 与 PHP 相 关 的 Internet 资 源 , 简要 讨论 了 
如 何 获 取 有 关 数 据 库 和 正文 未 讲述 的 主题 的 更 多 信息 ， 并 且 给 出 了 一 些 或 新 或 旧 的 表格 。 

除了 这 里 介绍 的 站 点 外 ， 还 应 该 知道 本 书 配套 网 站 的 网 址 一 一 www.LarryUllman.com。 在 这 里 
可 以 找到 本 书 用 到 的 所 有 脚本 ， 支 持 论坛 和 勘误 页 面 等 信息 。 

在 编写 本 书 第 一 版 时 ， 只 有 很 少 的 高 质量 PHP 站 点 ， 而 现在 则 有 差不多 十 儿 个 (还 有 数 以 百 
计 不 那么 好 的 ) PHP 站 点 。 最 好 的 、 最 显著 的 站 点 都 已 经 列 在 这 里 了 ,但 最 好 的 资源 永远 是 互联 
网 上 的 搜索 引擎 。 


B.1 PHP 在 线 资源 


如 果 有 特定 于 PHP 的 问题 ， 应 该 能 够 很 轻松 地 找到 答案 。 本 附录 的 这 一 节 主 要 介绍 可 供 使 用 
的 Internet 上 最 好 的 资源 。 





























B.1.1 PHP 手 册 


所 有 的 PHP 程 序 员 都 应 该 在 开始 使 用 这 门 语言 之 前 知道 并 且 已 经 获取 了 某 一 版 本 的 PHP 手 
册 。 从 PHP 官 方 网 站 (www.php.net/docs.php) 以 及 很 多 其 他 地 方 都 可 以 得 到 该 手册 。 

可 以 下 载 到 各 种 语言 、 各 种 格式 的 手册 。 官 方 网 站 还 提供 了 该 手册 的 注释 版 www.php.net 
manual/en/ (英文 ), 用 户 可 以 在 这 里 添加 有 用 的 注解 和 评论 。 如 果 对 某 个 特定 的 函数 不 是 很 了 解 ， 
阅读 手册 并 找到 该 函数 的 页 面 可 能 会 得 到 答案 。 

在 第 1 章 我 介绍 过 ， 通 过 www.php.net/functionname 可 以 快速 访问 为 任何 特定 函数 提供 的 文档 
页 面 。 例 如 ， 为 number_format () 函数 提供 的 页 面 是 www.php.net/number_format。 





























B.1.2 一 般 PHP 网 站 


本 市 提 到 了 在 编程 时 可 以 访问 的 众多 网 站 中 的 一 部 分 , 但 需要 自己 探索 才能 发 现 最 喜欢 的 网 
站 。 其 中 的 大 部 分 网 站 中 都 在 其 他 PHP 相 关 站 点 中 有 和 链接。 很 明显 ， 首 先 要 收藏 的 是 PHP.net 
(www.php.net) ， 这 是 PHP 的 官方 站 点 。 
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应 该 熟悉 Zend (www.zend.com) ， 这 是 PHP 核 心 创 建 者 的 主页 。 该 站 点 包含 了 很 多 可 下 载 的 
资源 和 其 他 大 量 的 资源 一 一 可 以 说 是 来 自 专 家 们 的 第 一 手 资料 。 

要 查找 特定 主题 的 信息 ，PHPBuilder (www.phpbuildercom) 是 一 个 不 错 的 站 点 。 该 站 点 有 
大 量 关于 如 何 使 用 PHP 来 完成 特定 任务 的 文章 。 此 外 ，PHPBuilder 还 提供 了 支持 论坛 和 代码 库 ， 
开发 者 可 以 在 这 里 上 传 示 例 脚本 。 

在 PHP Zone (http://php.dzone.com) 和 phplarchitect (www.phparch.com) 也 可 以 找到 很 有 用 
的 文章 ， 这 些 文章 也 经 常会 刊登 在 PHP 的 月 刊 杂志 中 。 

W3Schools (www.w3schools.com) 是 一 个 很 好 的 通用 Web 开 发 站 点 , 但 其 中 有 大 部 分 资源 是 
面向 PHP 的 。 从 使 用 PHP、HTML、CSS 和 JavaScript 开 发 动态 Web 站 点 的 角度 看 ， 这 是 一 个 非常 
理想 的 去 处 。 

如 果 我 发 现 了 好 的 资源 , 不 论 是 PHP 资 源 或 其 他 东西 , 我 一 般 会 将 它们 引用 在 自己 的 网 站 中 。 
你 可 以 在 我 的 网 站 的 PHP 目 录 下 找到 它们 ， 地 址 是 : www.larryullman.com/category/php/。 


B.1.3 ”代码 仓库 


如 今 在 线 的 代码 库 只 多 不 少 。 由 于 很 多 PHP 程 序 员 都 是 非常 慷慨 的 〈 而 且 通 常 乐于 分 享 )， 
因此 很 多 站 点 都 整理 了 大 量 可 供 下 载 的 PHP 脚 本 。 下 面 所 示 的 是 最 好 的 在 线 代 码 仓库 : 
口 HotScripts (www.hotscripts.com/PHP/) 。 























口 PHP Resource Index (http://php.resourceindex.com ) 。 
口 PHP Classes Repository (www.phpclasses.org ) 。 
口 PX: the PHP Code Exchange (http:/px.sklarcom ) 。 
浏览 Zend 和 PHPBuilder 或 搜索 Web 页 面 ， 也 能 找到 代码 示例 。 其 至 还 有 一 个 专门 用 于 查找 代 
码 的 搜索 引擎 www.koders.com 。 


B.1.4 新 闻 组 和 邮件 列表 


如 果 访 问 过 新 闻 组 ， 可 以 将 其 作为 强大 的 宣传 媒介 ,同时 它 还 可 以 找到 一 些 高 难度 问题 的 解 
决 方法 。 当 然 ， 也 可 以 向 这 些 新 闻 组 发 表 自己 的 专业 见解 ， 帮 助 那 些 有 这 方面 需要 的 人 。 

最 大 的 英语 PHP 新 闻 组 是 comp.lang.php。 可 以 通过 ISP 或 付费 的 Usenet 组 织 来 访问 它 。 新闻 组 
还 提供 除 英语 之 外 的 其 他 语言 支持 。 

PHP 网 站 列 出 了 可 供 注 册 的 邮件 列表 ， 位 于 www.php.net/mailing-lists.php。 
在 向 新 闻 组 或 邮件 列表 发 帖 之 前 ， 应 该 阅读 一 下 Eric Steven Raymond 所 写 的 “How to Ask 
Questions the Smart Way”, 参见 www.catb.org/~esr/faqs/ smart-questions.html。 花 十 分 钟 阅读 该 文档 ， 
便 可 以 在 寻求 帮助 时 为 你 节省 数 小 时 的 时 间 。 


B.2 数据库 资 源 
哪些 数据 库 资源 最 有 用 , 这 明显 取决 于 使 用 哪 种 数据 库 管 理 系 统 (DBMS ) 。 最 常 与 PHP 一 同 
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使 用 的 数据 库 可 能 就 是 MySQL 了 ， 但 PHP 能 够 支持 所 有 的 标准 数据 库 应 用 程序 。 

要 学 习 如 何 使 用 MySQL， 可 以 从 MySQL 的 官方 网 站 (www.mysql.com) 开始 。 可 以 下 载 一 
份 MySQL 和 手册 ， 作 为 工作 时 的 参考 资料 。 有 些 图 书 也 是 专门 介绍 MySQL 的 ， 包 括 我 自己 编著 的 
MySOL Visual QuickStart Guide, 2nd Edition (Peachpit Press, 2006 ) 。 

如 果 是 用 MySQL, 别 忘 了 下 载 并 安装 phpMyAdmin (www.phpmyadmin.net)。 这 是 一 款 用 PHP 
编写 的 ， 用 于 操作 数据 库 的 优秀 工具 。 如 果 使 用 的 是 PostgreSQL 或 者 其 至 是 Oracle， 也 可 以 找到 
类 似 的 工具 作为 接口 。 每 种 数据 库 应 用 程序 都 有 它 自己 的 邮件 列表 和 新 闻 组 。 

在 数据 库 领域 需要 钻研 的 另 一 块 资源 就 是 SQL。 下 面 列 出 了 讨论 SQL 的 网 站 ， 这 种 语言 会 
在 各 种 数据 库 应 用 程序 中 : 

D SQL Course (www.sqlcourse.com) 

D A Gentle Introduction to SQL (www.sqlzoo.net ) 

口 W3Schools’ SQL Tutorial (www.w3schools.com/sql) 
口 SQL.org (www.sql.org) 

我 编著 的 《PHP6 与 MySQL5 基 础 教程 》 一 书 也 讨论 了 SQL 和 MySQL， 并 且 比 本 书 讲述 得 更 
详细 。 



































B.3 Top 10 常见 问题 解答 


调试 是 一 种 很 有 用 的 技能 ,这 需要 花 时 间 并 具备 足够 的 经 验 才能 掌握 。 为 了 不 致 于 将 读者 带 
上 一 个 装备 不 良 的 征程 ， 本 书 将 列 出 PHP 脚 本 中 最 常见 的 10 种 问题 ,以 及 导致 这 些 问题 的 最 可 能 
的 原因 。 不 过 ， 首 先 本 书 会 给 出 调试 问题 时 的 5 个 最 佳 建议 。 

(1) 知道 正在 使 用 的 PHP 的 版 本 。 

有 些 问题 是 特定 于 PHP 的 某 个 版 本 的 。 

在 第 一 次 使 用 一 个 服务 器 时 , 请 使 用 phpinfo () 脚本 测试 所 使 用 的 版 本 。 另 外 还 要 确认 所 使 
用 的 MySQL 的 版 本 ， 如 果 可 以 的 话 ， 还 应 知道 操作 系统 、Web 服 务 器 (如 Apache 2.2) 等 。 

(2) 通过 URL 运 行 所 有 的 PHP 脚 本 。 

如 果 没 有 通过 URL 运 行 PHP 脚 本 (而 其 中 包含 了 向 PHP 脚 本 提交 表单 ),， 则 Web 服 务 器 不 会 处 
理 请 求 ， 这 意味 着 PHP 不 会 执行 任何 代码 。 

(3) 相信 错误 消息 ! 

很 多 初学 者 在 解决 问题 时 感到 非常 困难 ， 因 为 他 们 不 相信 所 看 到 的 错误 消息 。 尽 管 有 些 PHP 
错误 消息 很 隐 隐 甚 至 容易 让 人 误解 , 但 如 果 PHP 说 第 22 行 有 问题 , 那么 问题 很 可 能 就 是 在 第 22 行 。 

(4) 避免 通过 “尝试 ”来 解决 问题 ! 

如 果 不 能 确定 是 什么 原因 导致 了 问题 , 也 不 知道 确切 的 解决 方法 是 什么 , 请 避免 通过 随机 的 
尝试 来 解决 问题 。 这 样 很 可 能 会 导致 新 问题 的 产生 ， 并 且 只 会 进一步 搞 乱 原 有 的 问题 。 

(5) 休息 一 下 ! 

本 书 能 提供 的 最 佳 建议 就 是 离开 计算 机 休息 一 会 儿 , 我 通过 这 种 方式 解决 了 大 量 的 问题 。 很 
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多 时 候 真 正 需要 的 是 一 个 清晰 的 大 脑 。 
我 们 来 继续 ， 下 面 是 在 PHP 中 最 可 能 遇 到 的 10 个 问题 。 





(1) 空白 页 。 
如 果 在 提交 了 一 个 表单 或 加 载 了 一 个 PHP 脚 本 之 后 ， 在 Web 浏 览 器 看 到 的 是 空白 页 ， 则 很 可 
能 是 由 于 某 个 错误 中 断 了 页 面 的 执行 。 首 先 检查 HTML 源 代码 ， 看 看 是 否 有 HTML 错 误 。 然 后 打 


开 php . ini 配置 文件 或 PHP 脚 本 中 的 aisplay_errors， 看 看 出 现 了 什么 PHP 错 误 。 
(2) undefined variable 或 undefined index 错 误 (参见 图 B-1)。 








Notice: Undefined index: Name in /Users/larryullman/Sites 
/phpvqs4/handle_form.php on line 16 


Thank you. Mr. ,for your comments. 
图 B-1 对 由 拼写 或 大 小 写 错误 引发 的 未 定义 变量 或 索引 错误 的 解释 

只 有 当 错 误 报 告 设置 为 最 高 级 别 时 才 会 出 现 这 些 错 误 , 它们 可 能 指出 了 错误 , 也 可 能 根本 不 
是 错误 。 请 检查 每 个 变量 或 数组 索引 的 拼写 , 确保 它们 是 正确 的 。 然后, 或 者 修改 错误 报告 设置 ， 
或 者 在 引用 变量 之 前 对 其 初始 化 。 当 然 ， 还 要 确保 变量 的 确 带 有 值 。 

(3) 变量 不 带 有 值 。 

可 能 是 在 引用 变量 时 写 错 了 名 字 。 请 再 次 确认 变量 名 的 大 小 写 和 拼写 是 否 正确 , 然后 确保 正 
确 使 用 了 $_GET、$_POST、$_COOKIE 和 $_SESSION。 如 果 需 要 的 话 , 使 用 print_r () 函数 来 检 
查 每 个 变量 的 名 字 和 值 。 

(4) Call to undefined function... 错 误 。 

这 种 错误 消息 意味 着 试图 使 用 一 个 在 PHP 中 并 不 存在 的 函数 。 导 致 该 错误 产生 的 原因 可 能 是 
函数 名 拼写 错误 、 在 调用 函数 前 没有 定义 它 或 使 用 了 所 用 PHP 版 本 所 不 支持 的 函数 。 请 检查 拼写 
情况 ， 并 查看 PHP 手 册 中 关于 非 用 户 定义 函数 的 说 明 来 查找 问题 所 在 。 

(5) Headers already sent 错 误 (参见 图 B-2)。 

该 错误 消息 表明 在 Web 浏 览 器 已 经 接收 到 HTML 或 一 个 空白 页 之 后 调用 了 HTTP 头 相关 的 函 









































数 header () 、setcookie() 或 session_ start() 。 请 复查 在 调用 这 些 国 数 之 前 脚本 中 发 
生 了 什么 ， 或 使 用 输出 缓冲 来 避免 混乱 。 还 可 以 利用 输出 缓冲 来 避免 这 些 错误 的 出 现 。 
Warning: Cannot modify header information - headers already sent by loutput started 
at irUsersilarryullmanisSites!phpvagsd /templatesiheader .html:4) in FUsers/ilarryullman 
tSitesiphpvqs4ilogin.php on line 22 














图 B-2 茶 些 函数 因为 调用 时 机 错误 而 导致 的 headers already sent 错 误 


(6) Access denied 错 误 (参见 图 B-3)。 

如 果 在 尝试 使 用 数据 库 时 看 到 了 这 个 错误 消息 , 则 你 正在 使 用 的 用 户 名 、 密 码 和 主机 这 个 三 
元 组 可 能 不 具备 访问 数据 库 的 权限 。 这 通常 不 是 PHP 的 问题 。 请 确认 正在 使 用 的 值 ， 并 尝试 用 一 
个 不 同 的 系统 (如 MySQL 客 户 端 ) 去 连接 数据 库 。 
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Warning: mysqgl_connect() [function.mysqgl-connect]: Access denied for user 
userasme @ localhost (using password: YES) in 
/Users/larryullman/Sites/phpvqs4/mysql_connect.php on line 13 


Could not connect to MySQL 














图 B-3 ”如 果 MySQL 访 问 信息 不 正确 ， 则 会 看 到 数据 库 访问 被 拒绝 的 消息 














(7) Supplied argument is not a valid MySQL result resource 错 误 。 

这 是 另 一 个 数据 库 相 关 的 错误 消息 , 该 消息 意味 着 不 恰当 地 使 用 了 一 个 查询 结果 , 这 通常 是 
由 于 试图 在 一 个 没有 返回 任何 记录 的 查询 中 检索 行 。 要 解决 这 一 问题 ,请 找到 运行 的 查询 ， 并 使 
用 其 他 工具 〈 如 MySQL 客 户 端 或 phpMyAdmin) 测试 。 另 外 还 请 检查 变量 名 是 否 正确 。 

(8) 预 置 的 HTML 表 单 值 被 截断 。 

必须 将 HTML 表 单 中 文本 框 控件 的 value 属 性 值 放 置 在 双 引 号 中 。 如 果 不 这 么 做 ， 只 有 第 一 
个 空格 之 前 的 部 分 会 被 设置 为 文本 框 控 件 的 值 。 

(9) 条 件 或 循环 语句 产生 非 预 期 行为 。 

这 些 逻 辑 错误 非常 常见 。 请 确定 没有 用 错 运算 符 (如 错 将 == 写 成 =)， 以 及 引用 的 变量 确实 
正确 。 然 后 使 用 print 语 句 检查 脚本 做 了 些 什么 。 

(10) 解析 错误 (参见 图 B-4)。 

解析 错误 将 是 要 处 理 的 最 常见 的 错误 。 即 便 是 PHP 编 程 老 手 ， 也 会 偶尔 遇 到 这 个 问题 。 请 检 
查 每 条 语句 是 否 都 以 分 号 结尾 ， 以 及 所 有 的 引号 、 圆 括号 、 方 括号 和 花 括号 都 是 配对 的 。 如 有 果 还 
是 没 能 找到 解析 错误 ， 可 以 使 用 /* 和 */ 字 符 注 释 掉 大 段 代 码 。 然 后 每 次 取消 一 小 部 分 的 注释 ， 
直到 再 次 看 到 解析 错误 。 然 后 就 能 知道 脚本 中 哪 段 代 码 出 问题 了 (或 最 可 能 的 地 方 )。 
































Parse error: syntax error, unexpected 
T_ENCAPSED_AND_WHITESPACE. expecting T_STRING or 
T_VARIABLE or T_NUM_STRING in /Users/larryullman/Sites 
/phpvqs4/handle_form.php on line 19 


B-4 ”解析 错误 非常 常见 ， 而 且 能 阻止 脚本 的 执行 
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B.4 下 一 步 
本 书 只 是 帮助 你 开始 使 用 PHP， 但 在 这 之 后 ， 还 有 一 些 主题 需要 研究 。 
B.4.1 安全 


Web 服 务 器 、 操 作 系统 、 数 据 库 和 PHP 的 安全 这 些 主题 都 值得 拥有 各 自 的 书籍 。 尽 管 本 书 尝 
试 介绍 了 安全 Web 应 用 程序 的 编号， 但 这 个 领域 依然 有 很 多 区 域 是 需要 自己 去 探索 的 。 可 以 通过 
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下 面 这些 站 点 开始 你 的 探索 : 
口 Scarlet 学 习 (www.securereality.com.au/studyinscarlet.txt) ; 
口 W3C 安 全 资源 (www.w3.org/Security ) 。 

第 一 个 站 点 是 关于 一 篇 编写 有 关 安 全 PHP 代 码 的 文章 。 第 二 个 站 点 是 World Wide Web 
Consortium 针 对 Web 相 关 的 安全 性 问题 提供 的 资源 页 。 

还 需要 阅读 PHP 和 手册 和 所 使 用 的 数据 库 手 册 中 的 相关 章节 。 在 Internet 上 搜索 PHP 和 security 
也 会 找到 很 多 有 趣 的 文章 。 


B.4.2 面向 对 象 编程 


本 书 没有 涵盖 对 象 和 面向 对 象 编程 (OOP) 的 主题 ， 这 出 于 以 下 两 个 原因 : 

(1) 它 还 是 远 远 超过 了 初学 者 指南 的 范畴 ; 

(2) 不 理解 对 象 ， 并 不 会 限制 理解 PHP 能 够 做 什么 。 

如 果 决 定 要 学 习 这 一 主题 , 可 以 在 PHP 站 点 中 搜索 一 些 教程 , 找 一 个 框架 (参见 本 附录 的 B.4.3 
节 )， 或 阅读 一 下 我 编著 的 PHP 5 Advanced: Visual QuickPro Guide (Peachpit Press，2007) 。 我 在 
其 中 用 了 大 约 150 页 的 篇 幅 专 门 介绍 OOP (但 仍然 有 大 部 分 的 OOP 主 题 没 有 涉及 )1 























B.4.3 ”框架 


框架 是 一 种 已 经 建立 好 的 代码 库 ， 用 于 开发 成 熟 的 Web 应 用 程序 。 通 过 重用 已 经 由 其 他 人 证 
实 过 的 PHP 人 代码， 可 以 快速 地 建立 一 部 分 或 全 部 Web 站 点 。 

有 很 多 PHP 框 架 可 供 使 用 ， 可 以 从 PEAR 开 始 。PEAR 是 一 个 巨大 的 PHP 代 码 仓库 ， 它 是 使 用 
对 象 ( 从 技术 上 说 是 类 ) 编写 的 。 即便 自己 不 使 用 对 象 , 或 是 刚刚 理解 这 一 概念 ,依然 能 从 PEAR 
网 站 (http:// pear.php.net) 获得 大 量 有 价值 的 信息 。PEAR 及 其 Web 站 点 提供 了 免费 的 、 漂 亮 的 代 
码 并 展示 了 良好 的 PHP 编 码 风格 。PECL (http://pecl. php.net) 是 PEAR 的 更 为 强大 的 姊妹 库 。 

另 一 个 需要 考虑 的 框架 是 Zend Framework (http://framework.zend.com)。 这 个 框架 相对 较 新 ， 
具有 大 量 优点 ， 并 且 有 编写 完善 的 文档 。 

在 写作 本 书 时 ， 我 个 人 非常 喜欢 的 PHP 框 架 是 Yii (www.yiiframework.com) 。 在 我 的 网 站 中 我 
写 了 很 多 有 关 Yii 的 文章 。 

很 多 人 喜爱 框架 和 它们 提供 的 功能 。 但 另 一 方面 ， 学 习 使 用 一 个 框架 确实 要 花 一 些 时 间 ， 而 

且 自 定义 框架 的 行为 可 能 是 一 件 让 人 旦 惧 的 事情 。 
































B.4.4 JavaScript 和 Ajax 


JavaScript 是 运行 在 Web 浏 览 器 中 的 客户 端 技术 。 它 可 以 用 于 向 Web 站 点 增添 大 量 动态 特性 ， 
从 简单 的 诸如 图 片 翻 深 的 视觉 效果 ,到 交互 式 的 菜单 和 表单 。 由 于 JavaScript 是 运行 在 Web 浏 览 器 
中 的 ， 因 此 它 能 提供 一 些 PHP 所 不 能 提供 的 功能 。 不 过 ，JavaScript 和 PHP 一 样 ， 也 是 相当 易学 易 
用 的 。 更 多 信息 ， 请 参见 : 
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口 JavaScripticom (www.javascript.com ) ; 
口 W3School 上 的 JavaScript 页 (www.w3schools.com/js/)。 

Ajax (可 能 意味 着 异步 的 JavaScript 和 XML， 也 可 能 不 是 ， 取 决 于 问 的 是 谁 ) 自 2005 年 以 来 
已 经 成 了 Web 开 发 社区 里 最 热门 的 话题 。 这 项 技术 使 用 JavaScript 与 服务 器 进行 通信 , 但 用 户 并 不 
能 察觉 到 。 最 终 的 效果 是 ，Web 站 点 的 行为 越 来 越 像 桌面 应 用 程序 。 

更 多 信息 ， 请 参见 : Ajaxian (www.ajaxian.com); 或 在 网 络 上 搜索 相关 信息 。 

我 强烈 建议 你 学 习 一 下 jQuery (www.jquery.com)， 它 可 以 满足 你 对 JavaScript、Ajax 和 其 他 
动态 Web 的 需求 。jQuery 是 一 个 JavaScript 框 架 ， 非 常 容易 使 用 ， 功 能 也 非常 强大 ， 而 且 有 详细 的 
说 明文 档 。 


B.4.5 ”其 他 书籍 


我 希望 读者 在 读 完 本 书 之 后 ， 能 对 学 习 更 多 的 PHP 和 一 般 Web 开 发 知识 感 兴趣 。 尽 管 我 可 以 
推荐 其 他 作者 的 书 , 但 这 里 有 个 固有 的 冲突 , 而 且 我 认为 对 于 相同 的 读者 群 来 说 那些 作者 是 竞争 
对 手 。 因 此 ， 我 只 着 重 介绍 了 我 编著 的 其 他 几 本 书 ， 以 及 这 些 书 与 本 书 的 对 比 情况 。 

《PHP6 和 MySQL5 基 础 教程 》 是 本 书 的 后 续 (该 书 的 最 新 版 是 命名 为 PHP and MySQL for 
Dynamic Web Sites: Visual QuickPro Guide，Fourth Edition)。 两 本 书 有 些 内 容重 全 ， 尤 其 是 在 前 几 
章 ,， 但 它 与 本 书 使 用 的 例子 是 不 同 的 ， 而 且 比 本 书 的 讲述 步伐 更 快 。 特 别 是 涵盖 了 更 多 有 关 
MySQL 和 SQL 方 面 的 内 容 , 并 且 有 三 个 不 同 的 示例 章节 : 一 个 多 语言 论坛 、 一 个 用 户 注册 和 登录 
系统 以 及 一 个 电子 商务 站 点 。 

刚刚 提 到 过 ， 我 编著 的 PHP 5 Advanced: Visual QuickPro Guide (Peachpit Press, 2007) 也 是 本 
书 的 一 个 续集 。 这 本 书 的 内 容 更 为 高 级 ， 用 大 量 的 篇 幅 讲述 了 诸如 OOP 和 PEAR 这 样 的 主题 。 这 
本 书 并 不 需要 顺序 阅读 ， 这 与 本 书 不 同 ， 它 的 每 一 章 都 着 重 于 一 个 特定 的 主题 。 

MySOL: Visual QuickStart Guide, Second Edition (Peachpit Press, 2006) 这 本 书 看 上 去 是 专门 
讲解 MySQL 和 SQL 的 。 尽 管 其 中 有 4 章 涵盖 了 与 MySQL 交 互 的 语言 一 一 PHP、Perl 和 JavaScript， 
外 加 一 个 技术 章节 ， 但 这 本 书 还 是 主要 用 来 介绍 MySQL 的 安装 、 管 理 以 及 MySQL 知 识 。 

我 的 新 书 (在 写 这 本 书 之 前 完成 的 ) 是 Effortless E-Commerce with PHP and MySOL (New 
Riders, 2011)。 这 本 书 介绍 了 创建 功能 完备 的 电子 商务 网 站 所 需要 的 全 部 知识 。 书 中 使 用 了 两 个 
具体 的 示例 并 且 将 两 个 不 同 的 支付 系统 合并 在 一 起 。 当 然 ， 也 是 用 PHP 和 MySQL 实 现 的 。 

最 后 ， 我 编著 的 Building a Web Site with Ajax: Visual QuickProject Guide (Peachpit Press, 2008) 
一 书 介 绍 了 编写 一 个 Ajax 风 格 的 Web 站 点 的 全 过 程 。 这 本 书 也 使 用 了 PHP 和 MySQL, 但 没有 按照 
讲解 JavaScript 和 Ajax 的 方式 去 讲解 这 些 技 术 。 


B.5 表 






























































本 书 中 散布 着 不 少 表格 ， 为 了 便于 参考 ， 这 里 重新 列 出 


了 其 中 最 重要 的 3 个 表格 。 此 外 ， 还 
加 了 一 个 新 的 表格 , 列 出 了 运算 符 优先 级 ( 表 B-1)。 这 部 分 运算 符 是 


最 
按照 优先 级 从 高 到 低 列 出 的 ， 
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例如 ， 乘 法 的 优先 级 高 于 加 法 。 





表 B-1 运算 符 优先 级 
运算 符 优先 级 运算 符 优先 级 
! ++ —— [|| 
x pa 名 = += -= *= Ee 二 名 = 
+ 一 and 
0 Kor 
== != === or 
&& 








表 B-2 列 出 了 PHP 的 主要 运算 符 和 它们 的 类 型 。 最 重要 的 是 要 记得 单个 等 号 (=) 用 于 为 变 
赋值 ， 而 双 等 号 (==) 用 于 判断 相等 性 。 
表 B-2 ”PHP 运算 符 和 使 用 类 型 


且 - 
里 











运 算 符 用 法 类 型 
+ 加 算术 
减 算术 
/ 除 算术 
% 模 (除法 的 余数 ) 算术 
++ 增 量 算术 
-- 减 量 算术 
= 将 值 赋 给 变量 赋值 
== 相等 比较 
!= 不 相等 比较 
< 小 于 比较 
> 大 二 比较 
<= 小 于 或 等 于 比较 
>= 大 于 或 等 于 比较 
! 取 反 逻辑 
AND 与 逻辑 
&& 与 逻辑 
OR 或 逻辑 
|| 或 逻辑 

或 非 逻辑 
XOR 连接 字符 串 





表 B-3 指 出 了 在 打开 一 个 文件 时 可 以 使 用 的 模式 。 你 的 选择 决定 了 PHP 可 以 在 该 文件 上 进行 


的 操作 





写 入 、 读 取 等 。 











由 于 很 难 记 住 9ate () 函数 的 多 种 格式 ， 因 











此 在 使 用 date () 函数 时 ， 请 把 表 B-4 放 在 旁边 。 
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表 B-3 ”fopen() 的 模式 及 其 意义 








符 有 
只 读 ， 从 文件 的 起 始 位 置 开始 读 取 
读 或 写 ， 从 文件 的 起 始 位 置 开始 


或 

写 ， 如 果 文 件 不 存在 则 创建 文件 ， 并 且 盖 写 现 有 内 容 

或 写 ， 如 果 文 件 不 存在 则 创建 文件 ， 并 且 盖 写 现 有 内 容 ( 写 入 时 ) 
写 

或 























; 如 果 文 件 不 存在 则 创建 文件 ， 将 新 数据 追加 到 文件 末尾 (保留 任何 现 有 数据 并 向 其 中 添加 新 数据 ) 
写 ， 如 果 文 件 不 存在 则 创建 文件 ， 将 新 数据 追加 到 











注 总 党 并 








文件 末尾 ( 写 入 时 ) 

















只 写 ， 如果 文 件 不 存在 则 创建 文件 ， 如 果 文 件 存 在 则 什么 也 不 做 〈 并 发 出 一 个 警告 ) 
读 或 写 ， 如 果 文 件 不 存在 则 创建 文件 ， 如 果 文 件 存在 则 什么 也 不 做 〈 并 发 出 一 个 警告 ) ( 写 和 时 ) 



































表 B-4 aate() 函数 的 格式 化 字符 及 其 意义 与 示例 




















































































































字 符 含义 示 例 
Y 4 位 数字 表示 的 年 份 2011 
y 2 位 数字 表示 的 年 份 11 
L 是 否 为 国 年 1 (表示 “是 ”) 
n 1 或 2 位 数字 表示 的 月 份 2 
m 2 位 数字 表示 的 月 份 02 
F 月 份 February 
M 3 个 字母 表示 的 月 份 Feb 
1 或 2 位 数字 表示 的 月 份 中 的 一 天 8 
2 位 数字 表示 的 月 份 中 的 一 天 08 
1 (小 写 的 LL) 一 周 中 的 某 一 天 Monday 
D 3 个 字母 表示 的 一 周 中 的 某 一 天 Mon 
w 1 位 数字 表示 的 一 周 中 的 某 一 天 0 (星期 日 ) 
z 一 年 中 的 某 一 天 : 0~365 189 
t 月 份 中 有 多 少 天 31 
2 个 字符 表示 的 天 数 英文 序数 词 后 级 rd 
g 小 时 数 ，1 或 2 位 数字 表示 的 12 小 时 制 格式 6 
G 小 时 数 ，1 或 2 位 数字 表示 的 24 小 时 制 格式 18 
h 小 时 数 ，2 位 数字 表示 的 12 小 时 制 格式 06 
H 小 时 数 ，2 位 数字 表示 的 24 小 时 制 格式 18 
i 分 钟 数 45 
S 秒 数 18 
u 毫秒 数 1234 
a am 或 pm am 
A AM 或 PM PM 
U 从 epoch 开 始 的 秒 数 1048623008 
e 时 区 UTC 
I (大 写 的 i) 是 否 为 夏令 时 1 (表示 “是 ”) 


O 与 GMT 之 间 的 时 差 +0600 


x 


了 i 站 在 巨人 的 让 上 
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“Larry UllIman 写 的 这 本 PHP 书 体现 他 两 个 重要 的 才能 : ( 1 ) 他 能 够 让 你 对 PHP 有 全 面 的 了 解 ; ( 2 ) 他 能 够 用 简明 的 语言 
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一 一 资深 计算 机 取证 专家 Jerry Saperstein 


“我 的 PHP 知 识 很 有 限 ， 通 读 完 这 本 书 又 做 了 很 多 书 上 的 练习 之 后 ， 我 学 会 了 。 至 少 对 我 来 说 ， 这 本 书 实现 了 其 承诺 ， 即 快 
速 、 便 捷 地 学 会 PHP。” 
一 一 硅谷 高 级 管理 人 员 和 技术 顾问 Bruce B. Razban 
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